成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

C#基礎(chǔ):理解裝箱與拆箱

開發(fā) 后端
前面我們講到 .NET 平臺支持的兩大數(shù)據(jù)類型:值類型和引用類型。值類型比引用類型更高效,因?yàn)樗鼪]有指針引用,不用分配在托管堆中,也不用被 GC 回收。

[[422064]]

本文轉(zhuǎn)載自微信公眾號「精致碼農(nóng)」,作者liamwang。轉(zhuǎn)載本文請聯(lián)系精致碼農(nóng)公眾號。

前面我們講到 .NET 平臺支持的兩大數(shù)據(jù)類型:值類型和引用類型。值類型比引用類型更高效,因?yàn)樗鼪]有指針引用,不用分配在托管堆中,也不用被 GC 回收。但有時候你可能偶爾需要將一種類型的變量表示為另一種類型的變量。為此,C# 提供了裝箱和拆箱的機(jī)制。

1理解裝箱

簡單地說,裝箱就是將一個值類型的數(shù)據(jù)存儲在一個引用類型的變量中。

假設(shè)你一個方法中創(chuàng)建了一個 int 類型的本地變量,你要將這個值類型表示為一個引用類型,那么就表示你對這個值進(jìn)行了裝箱操作,如下所示:

  1. static void SimpleBox() 
  2.   int myInt = 25; 
  3.  
  4.   // 裝箱操作 
  5.   object boxedInt = myInt; 

確切地說,裝箱的過程就是將一個值類型分配給 Object 類型變量的過程。當(dāng)你裝箱一個值時,CoreCLR 會在堆上分配一個新的對象,并將該值類型的值復(fù)制到該對象實(shí)例。返回給你的是一個在托管堆中新分配的對象的引用。

2理解拆箱

反過來,將 Object 引用類型變量的值轉(zhuǎn)換回棧中相應(yīng)的值類型的過程則稱為拆箱。

從語法上講,拆箱操作看起來就像一個正常的轉(zhuǎn)換操作。然而,其語義是完全不同的。CoreCLR 首先驗(yàn)證接收的數(shù)據(jù)類型是否等同于被裝箱的類型,如果是,它就把值復(fù)制回基于棧存儲的本地變量中。

例如,如果下面的 boxedInt 的底層類型確實(shí)是 int,那就完成了拆箱操作:

  1. static void SimpleBoxUnbox() 
  2.   int myInt = 25; 
  3.  
  4.   // 裝箱操作 
  5.   object boxedInt = myInt; 
  6.  
  7.   // 拆箱操作 
  8.   int unboxedInt = (int)boxedInt; 

記住,與執(zhí)行典型的類型轉(zhuǎn)換不同,你必須將其拆箱到一個恰當(dāng)?shù)臄?shù)據(jù)類型中。如果你試圖將一塊數(shù)據(jù)拆箱到不正確的數(shù)據(jù)類型中,將會拋出 InvalidCastException 異常。為了安全起見,如果你不能保證 Object 類型背后的類型,最好使用 try/catch 邏輯把拆箱操作包起來,盡管這樣會有些麻煩。考慮下面的代碼,它將拋出一個錯誤,因?yàn)槟阏噲D將裝箱的 int 類型拆箱成一個 long 類型:

  1. static void SimpleBoxUnbox() 
  2.   int myInt = 25; 
  3.  
  4.   // 裝箱操作 
  5.   object boxedInt = myInt; 
  6.  
  7.   // 拆箱到錯誤的數(shù)據(jù)類型,將觸發(fā)運(yùn)行時異常 
  8.   try 
  9.   { 
  10.     long unboxedLong = (long)boxedInt; 
  11.   } 
  12.   catch (InvalidCastException ex) 
  13.   { 
  14.     Console.WriteLine(ex.Message); 
  15.   } 

3生成的 IL 代碼

當(dāng) C# 編譯器遇到裝箱/拆箱語法時,它會生成包含裝箱/拆箱操作的 IL 代碼。如果你用 ildasm.exe 查看編譯的程序集,你會看到裝箱和拆箱操作對應(yīng)的 box 和 unbox 指令:

  1. .method assembly hidebysig static 
  2.     void  '<<Main>$>g__SimpleBoxUnbox|0_0'() cil managed 
  3.   .maxstack  1 
  4.   .locals init (int32 V_0, object V_1, int32 V_2) 
  5.     IL_0000:  nop 
  6.     IL_0001:  ldc.i4.s   25 
  7.     IL_0003:  stloc.0 
  8.     IL_0004:  ldloc.0 
  9.     IL_0005:  box        [System.Runtime]System.Int32 
  10.     IL_000a:  stloc.1 
  11.     IL_000b:  ldloc.1 
  12.     IL_000c:  unbox.any  [System.Runtime]System.Int32 
  13.     IL_0011:  stloc.2 
  14.     IL_0012:  ret 
  15.   } // end of method '<Program>$'::'<<Main>$>g__SimpleBoxUnbox|0_0' 

 

 

乍一看,裝箱/拆箱似乎是一個沒啥用的語言特性,學(xué)術(shù)性大于實(shí)用性。畢竟,你很少需要在一個本地 Object 變量中存儲一個本地值類型。然而,事實(shí)是裝箱/解箱過程是相當(dāng)有用的,因?yàn)樗试S你假設(shè)一切都可以被當(dāng)作 Object 類型來處理,而 CoreCLR 會自動幫你處理與內(nèi)存有關(guān)的細(xì)節(jié)。

4實(shí)際應(yīng)用

讓我們來看看裝箱/拆箱的實(shí)際應(yīng)用,我們以 C# 的 ArrayList 類為例,用它來保存一批在棧中存儲的整型數(shù)據(jù)。ArrayList 類的相關(guān)方法成員列舉如下:

  1. public class ArrayList : IList, ICloneable 
  2.   ... 
  3.   public virtual int Add(object? value); 
  4.   public virtual void Insert(int index, object? value); 
  5.   public virtual void Remove(object? obj); 
  6.   public virtual object? this[int index] { get; set; } 

請注意,上面 ArrayList 的方法都是對 Object 類型數(shù)據(jù)進(jìn)行操作。ArrayList 是為操作對象(代表任何類型)而設(shè)計(jì)的,而對象是在托管堆上分配的數(shù)據(jù)。請考慮下面代碼:

  1. static void WorkWithArrayList() 
  2.   // 當(dāng)傳遞給對象的方法時,值類型會自動被裝箱 
  3.   ArrayList myInts = new ArrayList(); 
  4.   myInts.Add(10); 

盡管你直接將數(shù)字?jǐn)?shù)據(jù)傳入需要 Object 參數(shù)的方法中,但運(yùn)行時自動將分配在棧中的數(shù)據(jù)裝箱。如果你想使用索引器從 ArrayList 中檢索一條數(shù)據(jù),你必須使用轉(zhuǎn)換操作將堆分配的對象拆箱為棧分配的整型,因?yàn)?ArrayList 的索引器返回的是 Object 類型,而不是 int 類型。

  1. static void WorkWithArrayList() 
  2.   // 當(dāng)傳遞給需要對象參數(shù)的方法時,值類型就自動被裝箱 
  3.   ArrayList myInts = new ArrayList(); 
  4.   myInts.Add(10); 
  5.  
  6.   // 當(dāng)對象被轉(zhuǎn)換回基于棧存儲的數(shù)據(jù)時,就會發(fā)生拆箱 
  7.   int i = (int)myInts[0]; 
  8.  
  9.   // 由于 WriteLine() 需要的 object 參數(shù),又重新裝箱了 
  10.   Console.WriteLine("Value of your int: {0}", i); 

在調(diào)用 ArrayList.Add() 之前,在棧中分配的 int 數(shù)值被裝箱了,所以它可以被傳入?yún)?shù)為 Object 類型的方法中。從 ArrayList 中檢索到 Object 類型的數(shù)據(jù)時,通過轉(zhuǎn)換操作,它就被拆箱成 int 類型。最后,當(dāng)它被傳遞給 Console.WriteLine() 方法時,又被裝箱了,因?yàn)檫@個方法的參數(shù)是 Object 類型。

5小結(jié)

從程序員的角度來看,裝箱和拆箱是很方便的,我們不需要手動去復(fù)制和轉(zhuǎn)移內(nèi)存中的值類型和引用類型的數(shù)據(jù)。

但裝箱和拆箱背后的棧/堆內(nèi)存轉(zhuǎn)移也帶來了性能問題。下面總結(jié)一下對一個簡單的整型數(shù)進(jìn)行裝箱和拆箱所需要的步驟:

在托管堆中分配一個新對象;

在棧中的數(shù)據(jù)值被轉(zhuǎn)移到該托管堆中的對象上;

當(dāng)拆箱時,存儲在堆中對象上的值被轉(zhuǎn)移回棧中;

堆上未使用的對象將最終被 GC 回收。

盡管很多時候裝箱和拆箱操作不會在性能方面造成重大影響,但如果一個像 ArrayList 這樣的集合包含成千上萬條數(shù)據(jù),而你的程序又會頻繁操作這些數(shù)據(jù),性能的影響還是會很明顯的。

所以,我們平時在編程時應(yīng)當(dāng)盡量避免發(fā)生裝箱和拆箱操作。比如對于上面 ArrayList 的示例,如果集合元素類型是一致的,則應(yīng)當(dāng)使用泛型的集合類型,比如改用 List、LinkedList 等。

 

 

 

責(zé)任編輯:武曉燕 來源: 精致碼農(nóng)
相關(guān)推薦

2009-08-26 03:39:00

C#裝箱和拆箱

2009-09-01 17:51:47

C#拆箱C#裝箱

2009-08-28 11:22:11

C#裝箱和拆箱

2009-08-06 15:40:11

C#裝箱和拆箱

2020-11-02 13:06:42

Java裝箱拆箱

2009-10-22 19:11:25

CLR Via C#教

2015-09-02 10:12:54

Java自動裝箱拆箱

2012-03-26 11:32:45

Java

2009-08-11 15:17:12

C#基礎(chǔ)知識

2024-05-17 12:56:09

C#編程線程

2009-08-13 18:02:50

C#基礎(chǔ)概念

2009-08-27 11:43:31

C#語法

2024-12-26 00:28:59

C#base?關(guān)鍵字

2009-08-11 14:20:41

C# .NET學(xué)習(xí)經(jīng)驗(yàn)

2019-10-30 16:03:48

JavaJava虛擬機(jī)數(shù)據(jù)庫

2009-08-24 14:26:42

C# 泛型類

2010-05-04 08:58:02

.NET

2009-08-27 16:37:06

C#基礎(chǔ)知識

2009-09-03 16:51:27

C#類屬性

2009-09-17 17:44:51

C#動態(tài)數(shù)組
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 欧美日韩国产精品 | 午夜精品一区二区三区在线观看 | 亚洲国产二区 | 亚洲九色| 亚洲午夜视频在线观看 | 狠狠av| 97免费在线观看视频 | 成人不卡 | 成人午夜在线 | 国产精品国产精品国产专区不卡 | 91在线中文字幕 | 日韩在线资源 | av在线视| 久久久网 | 人人九九精 | 九九九国产| 久久国产一区 | 亚洲一区二区精品视频 | 亚洲成人免费电影 | 曰韩三级| 日韩在线三级 | 国产免费拔擦拔擦8x高清 | 国产精品视频综合 | 中文字幕在线视频观看 | 高清一区二区视频 | 久久中文字幕一区 | 国产精品一二三区在线观看 | 成人免费看黄 | 中文字幕成人网 | 国产免费一区 | 999精品视频 | 中文字幕av网 | 国产日韩电影 | av喷水| 午夜久久av| va精品 | 久久久久久艹 | 国产精品国产a级 | 天天干天天插 | 久久国产精品无码网站 | 少妇精品亚洲一区二区成人 |