關(guān)于C#?dynamic裝箱問題
前言
前幾天在技術(shù)群里看到有同學(xué)在討論關(guān)于dynamic是否會存在裝箱拆箱的問題,我當時第一想法是"會"。至于為啥會有很多人有這種疑問,主要是因為覺得dynamic
可能是因為有點特殊,因為它被稱為動態(tài)類型
,可能是因為這里的動態(tài)對大家造成的誤解,認為這里的動態(tài)可以推斷出具體的類型,所以可以避免裝箱拆箱。但是事實并不是這樣,今天就一起就這個問題雖然討論一下。
裝箱拆箱
首先咱們先來看下何為裝箱拆箱,這個可以在微軟官方文檔中Boxing and Unboxing文檔中看到答案,咱們就簡單的摘要一下相關(guān)的描述
裝箱是將值類型轉(zhuǎn)換為類型對象或此值類型實現(xiàn)的任何接口類型的過程。當公共語言運行時 (CLR) 將值類型裝箱時,它會將值包裝在 System.Object 實例中并將其存儲在托管堆上。拆箱從對象中提取值類型。拳擊是隱含的;拆箱是明確的。裝箱和拆箱的概念是 C# 類型系統(tǒng)統(tǒng)一視圖的基礎(chǔ),其中任何類型的值都可以視為對象。
翻譯起來會比較抽象,理解起來就是利用裝箱和拆箱功能,可通過允許值類型的任何值與Object 類型的值相互轉(zhuǎn)換,將值類型與引用類型鏈接起來。也就是值類型和引用類型相互轉(zhuǎn)換的一做橋梁,但是問題也很明顯那就是實例會存在在堆棧之前相互copy的問題,會存在一定的性能問題,所以這也一直是一個詬病。
雖然說是這樣但是也沒必要一直扣死角,畢竟很多時候程序還沒有糾結(jié)到這種程度,因為任何語言存在的各種方法中或者操作中都會有一定這種問題,所以本質(zhì)不是語言存在各種問題,而是在什么場景如何使用的問題。比如避免出現(xiàn)裝箱和拆箱的辦法也就是入概念所說的,那就是避免值類型和和引用類型之間相互轉(zhuǎn)換,但是很多時候還是避免不了的,所以也不必糾結(jié)。
探究本質(zhì)
上面講解了關(guān)于裝箱拆箱
的概念,接下來咱們就來定義一段代碼看看效果,為了方便對比咱們直接對比著看一下
dynamic num = 123; dynamic str = "a string";
想要看清本質(zhì)還是要反編譯一下生成的結(jié)果看一下的,這里我們可以借助ILSpy
或dnSpy
來看下,首先看一下反編譯回來的效果
private static void <Main>$(string[] args) { object num = 123; object str = "a string"; Console.ReadKey(); }
因為我是使用的是.net6的頂級聲明方式所以會生成<Main>$
方法。不過從反編譯的結(jié)果就可以看出來dynamic的本質(zhì)是object
,如果還有點懷疑的話可以直接查看生成的IL
代碼,還是使用ILSpy
工具
.method private hidebysig static void '<Main>$' ( string[] args ) cil managed { // Method begins at RVA 0x2094 // Header size: 12 // Code size: 30 (0x1e) .maxstack 1 .entrypoint .locals init ( // 這里可以看出聲明的num和str變量都是object類型的 [0] object num, [1] object str ) // object obj = 123; IL_0000: ldc.i4.s 123 // 這里的box說明存在裝箱操作 IL_0002: box [System.Runtime]System.Int32 IL_0007: stloc.0 // object obj2 = "a string"; IL_0008: ldstr "a string" IL_000d: stloc.1 // Console.ReadKey(); IL_000e: call valuetype [System.Console]System.ConsoleKeyInfo [System.Console]System.Console::ReadKey() IL_0013: pop // (no C# code) IL_0014: nop IL_0015: nop IL_0016: nop IL_0017: nop IL_0018: nop IL_0019: nop IL_001a: nop IL_001b: nop // } IL_001c: nop IL_001d: ret } // end of method Program::'<Main>$'
通過這里可以看出dynamic的本質(zhì)確實是object
,既然是object那就可以證實確實是存在裝箱操作。這個其實在微軟官方文檔Using type dynamic上有說明,大致描述是這樣的
dynamic類型是一種靜態(tài)類型,但類型為dynamic的對象會跳過靜態(tài)類型檢查。大多數(shù)情況下,該對象就像具有類型object一樣。 在編譯時,將假定類型化為dynamic的元素支持任何操作。因此,不必考慮對象是從 COM API、從動態(tài)語言(例如 IronPython)、從 HTML 文檔對象模型 (DOM)、從反射還是從程序中的其他位置獲取自己的值。但是,如果代碼無效,則在運行時會捕獲到錯誤。
從這里可以看出dynamic表現(xiàn)出來的就是object,只是dynamic會跳過靜態(tài)類型檢查,所以編譯的時候不會報錯,有錯誤的話會在運行的時候報錯,也就是我們說的是在運行時確定具體操作。這涉及到動態(tài)語言運行時,動態(tài)語言運行時(DLR)是一種運行時環(huán)境,可以將一組動態(tài)語言服務(wù)添加到公共語言運行時(CLR)。使用DLR可以輕松開發(fā)在.NET上運行的動態(tài)語言,并為靜態(tài)類型語言添加動態(tài)特征。
匿名類型
總會有人拿dynamic
和var
進行比較,但是本質(zhì)上來說,這兩者描述的不是一個層面的東西。var
叫隱式類型,本質(zhì)是一種語法糖,也就是說在編譯的時候就可以確定類型的具體類型,也就是說var本質(zhì)是提供了一種更簡單的編程體驗,不會影響變量本身的行為。
這也就解釋了為啥同一個var
變量多次賦值不能賦不同類型的值,比如以下操作編譯器會直接報錯
var num = 123; num = "123"; //報錯
如果你是用的集成開發(fā)環(huán)境的話其實很容易發(fā)現(xiàn),把鼠標放到var
類型上就會顯示變量對應(yīng)的真實類型?;蛘呖梢灾苯油ㄟ^ILSpy
看看反編譯結(jié)果,比如聲明了var num = 123
編譯完成之后就是
private static void <Main>$(string[] args) { int num = 123; Console.ReadKey(); }
請注意這里并不是object
而是轉(zhuǎn)換成了具體的類型因為123
就是int類型的,嚴謹一點看一下IL
代碼
.maxstack 1 .entrypoint //聲明的int32 .locals init ( [0] int32 num ) // int num = 123; IL_0000: ldc.i4.s 123 IL_0002: stloc.0
相信這里就可以看出來了dynamic
和var
確實也不是一個層面的東西。var是隱式類型是語法糖為了簡化編程體驗用的,dynamic則是動態(tài)語言運行時技術(shù),編譯時轉(zhuǎn)換成object類型,因為在c#上一切都是object,然后再運行時進行具體的操作。
總結(jié)
本篇文章主要是在技術(shù)群里看到有同學(xué)在討論關(guān)于dynamic是否會裝箱引發(fā)的思考,相對來說講解的比較基礎(chǔ)也比較簡單。想對一個東西理解的更透徹,就要一步一步的了解它到底是什么,這樣的話就可以更好的理解和思考。也印證了那句話,你不會用或者用是因為你對它不夠了解,當你對它有足夠理解的時候,操作起來也就會游刃有余。
到此這篇關(guān)于關(guān)于C# dynamic裝箱引發(fā)的思考的文章就介紹到這了,更多相關(guān)C# dynamic裝箱內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
【C#基礎(chǔ)】Substring截取字符串的方法小結(jié)(推薦)
這篇文章主要介紹了Substring截取字符串方法小結(jié),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-05-05C#?Socket數(shù)據(jù)接收的三種實現(xiàn)方式
本文主要介紹了C#?Socket數(shù)據(jù)接收的三種實現(xiàn)方式,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-07-07C#實現(xiàn)數(shù)組元素的數(shù)據(jù)類型轉(zhuǎn)換方法詳解
這篇文章主要為大家介紹了C#中一維數(shù)組如何快速實現(xiàn)數(shù)組元素的數(shù)據(jù)類型的轉(zhuǎn)換,文中的示例代碼講解詳細,感興趣的小伙伴可以了解一下2022-04-04C#開發(fā)Windows UWP系列之布局面板RelativePanel
這篇文章介紹了C#開發(fā)Windows UWP系列之布局面板RelativePanel,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-06-06