當(dāng)前位置:首頁 > 芯聞號 > 充電吧
[導(dǎo)讀]我剛用C#一個來月,可能理解得不對,還請大家指教。讀懂文章你需要對C語言的指針有所理解。需要注意區(qū)別:對C\C++來說,任何類型都可以當(dāng)成C#的“引用類型”,因為有指針?!驹趦?nèi)存上】void?foo(

我剛用C#一個來月,可能理解得不對,還請大家指教。
讀懂文章你需要對C語言的指針有所理解。

需要注意區(qū)別:對CC++來說,任何類型都可以當(dāng)成C#的“引用類型”,因為有指針。

【在內(nèi)存上】

void?foo()
{
????int?aaa?=?0;?//值類型,aaa在Stack上分配(SUB?ESP,XX)
????int*?paaa?=?new?int[123];?//引用類型,paaa在Heap上分配,HeapAlloc(GetProcessHeap()...)
????foo2(&aaa);?//【引用】值類型aaa
}->函數(shù)退出,aaa從棧上銷毀,paaa依然存在在進(jìn)程堆中。

void?foo2(int*?paaa)
{
????*paaa?=?123;
}


【Box和UnBox】
打個很簡單的比方:
有函數(shù):

void?foo(int*?pa)
{
????beginthread(&new_thread...,pa);?//啟動新線程操作指針pa
}

void?new_thread(int*?pa)
{
????Sleep(1000);?//新線程睡眠1秒
????if?(*pa?==?123)?//判斷pa指針的內(nèi)容是123
????????OutputDebugStringA("OK.");????
}


還有函數(shù):

void?exec1()
{
????int?pa?=?123;
????foo(&pa);
}



大家可能看出問題了,當(dāng)我們執(zhí)行foo指向棧上的pa的時候,因為exec1函數(shù)已經(jīng)退出,pa的值被系統(tǒng)回收,不可預(yù)料,線程體在睡眠一秒后去訪問這個棧上的地址可能得到的值不是123,但是如果我們在堆上分配:

void?exec2()
{
????int*?pa?=?(int*)malloc(sizeof(int));
????*pa?=?123;
????foo(pa);
}


執(zhí)行exec2,就可以確保pa不會被回收,因為沒有顯式free。

C#同理,Box就是把在exec1中的值類型pa進(jìn)行exec2的處理,然后執(zhí)行處理后pa的值在exec1退出后依然在棧上銷毀,但是執(zhí)行foo給進(jìn)去的pa是一個.NET通過類似malloc這種分配堆內(nèi)存的方法分配的一個新的指針,并且把原來棧上的值pa給memcpy進(jìn)去了,并且把這個指針交給了foo函數(shù),這個【指針】就是被裝箱(Box)后的【引用類型】,此時foo引用了pa,并且把它又交給new_thread,new_thread操作完了pa后,再也沒人使用這個引用類型pa,GC會把pa進(jìn)行free。

我們可以得出,棧類型在函數(shù)體退出后,會被系統(tǒng)自動釋放,而堆類型不會,還記得ATL的CComPtr么,而堆數(shù)據(jù)則要靠GC進(jìn)行垃圾回收。
如果你對Windows COM和C++有理解,可以這樣想象:
ATL::CComPtr = 值類型auto-free。為什么,因為CComPtr在函數(shù)體內(nèi)構(gòu)造(AddRef),在函數(shù)體退出析構(gòu)(Release)。
std::shared_ptr = 引用類型auto-free。類是new來的,shared_ptr進(jìn)行引用計數(shù)管理,在某個函數(shù)體退出時引用計數(shù)跑0,才進(jìn)行delete。

然后我們還可以得出,值類型指向?qū)嶋H的數(shù)據(jù),而引用類型指向數(shù)據(jù)指針。
這個你用后腦就可以想出來了,例子:

void?foo()
{
????int?aa?=?123;?//aa直接指向123,編譯器通過ESP指針進(jìn)行間接訪問
????int*?paa?=?(int*)malloc(sizeof(int));?//paa指向堆上的一塊內(nèi)存
????*paa?=?123;?//編譯器使用直接指針訪問
}


所以說,C#中,值類型還是棧變量,引用類型則是shared_ptr。

反之,引用到值的拆箱:

void?foo()
{
????int*?pa?=?new?int[123];?//引用類型
????pa[0]?=?123;

????int?aa?=?*pa;?//aa是棧上的值類型,從堆上把數(shù)據(jù)進(jìn)行memcpy,UnBox
}



所以這2個過程是即浪費時間又浪費內(nèi)存的。
從上面我們可以看出,值類型裝箱到引用類型,裝箱后,值類型改變了,被裝箱的引用類型不會改變,反之同理,比如:

void?foo()
{
????int?aa?=?123;?//值類型
????int*?pa?=?new?int[123];?//引用類型
????pa[0]?=?aa;?//堆上的引用類型數(shù)據(jù)改了

????aa?=?456;?//棧上的值類型變了,但堆上的數(shù)據(jù)可不會同步變,這可不是WPF的數(shù)據(jù)綁定,哈哈
}


用C#代碼:

private?void?Form1_Load(object?sender,?EventArgs?e)
{
????????????int?aaa?=?123;?//Value?Type
????????????IntPtr?paaa?=?System.Runtime.InteropServices.Marshal.AllocHGlobal(sizeof(int));?//Ref?Type

????????????System.Runtime.InteropServices.Marshal.WriteInt32(paaa,aaa);?//Box
????????????aaa?=?System.Runtime.InteropServices.Marshal.ReadInt32(paaa);?//UnBox
}


【繼承關(guān)系】
C#的玩意兒都繼承System.Object,棧上的值類型繼承System.ValueType,這是編譯器定義的隱性繼承。
也就是這樣:

void?foo()
{
????int?aaa?=?123;
????System.Diagnostics.Debug.WriteLine(((aaa?as?ValueType)?as?object).ToString());
}


于是可以這樣:

private?bool?IsValueType(object?o)
{
????????????return?o?is?ValueType???true?:?false;
}

private?void?Form1_Load(object?sender,?EventArgs?e)
{
????????????int?aaa?=?123;
????????????System.Diagnostics.Debug.WriteLine(IsValueType(aaa).ToString());?//True

????????????var?bc?=?new?Form();
????????????System.Diagnostics.Debug.WriteLine(IsValueType(bc).ToString());?//False
}



【好玩的東西】
值類型的引用類型:
大家知道,值類型使用棧內(nèi)存,在C++中,我們有棧上類(就是RIIA,資源分配即初始化),但是在C#中,所有類都必需使用new進(jìn)行堆上申請,由GC進(jìn)行統(tǒng)一管理。
比如在DirectShow基本類里面的自動臨界區(qū)類CCritSec,就像C#里面的lock關(guān)鍵字一樣。

棧的特征就是分配->初始化:
int aa = 123;
退出自動銷毀。

這個RIIA范式在C++程序員手里那是用得爽。
C#下,因為一切引用類型都是在堆上跑,我們?nèi)绻鱿馬IIA這樣的東西怎么辦?
可能你已經(jīng)想到了,using關(guān)鍵字和IDisposable。

這個我就不多說了,自己搜索一下就知道了。

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫毥谦F公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運行,同時企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險,如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點: 有效應(yīng)對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅持高質(zhì)量發(fā)展策略,塑強核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運營商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術(shù)學(xué)會聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(shù)(集團(tuán))股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉