C# 封箱和拆箱
封箱(boxing)是把值類型轉(zhuǎn)換為System.Object類型,或者轉(zhuǎn)換為由值類型實現(xiàn)的接口類型。拆箱(unboxing)是相反的轉(zhuǎn)換過程。
例如,下面的結(jié)構(gòu)類型:
struct MyStruct
{
public int Val;
}
可以把這種類型的結(jié)構(gòu)放在object類型的變量中,對其封箱:
MyStruct valTypel = new MyStruct();
valTypel.Val = 5;
object refType = valTypel?
其中創(chuàng)建了一個類型為MyStruct的新變量valTypel,并把一個值賦予這個結(jié)構(gòu)的Val成員,然后把它封箱在object類型的變量refType中。
以這種方式封箱變量而創(chuàng)建的對象,會包含值類型變量的一個副本的引用,而不包含源值類型變量的引用。要進行驗證,可以修改源結(jié)構(gòu)的內(nèi)容,然后把對象中包含的結(jié)構(gòu)拆箱到新變量中,并檢査其內(nèi)容:
valTypel.Val = 6;
MyStruct valType2 = (MyStruct)refType;
WriteLine {$uvalType2 .Val = {valType2 .Val}");
執(zhí)行這段代碼將得到如下輸出結(jié)果:
valType2.Val = 5
但在把一個引用類型賦予對象時,將執(zhí)行不同的操作。把MyStruct改為一個類(不考慮這個類名不合適的情況),即可看到這種情形:
class MyStruct
{
public int Val;
}
如果不修改上面的客戶代碼(再次忽略名稱有誤的變量),就會得到如下輸出結(jié)果:
valType2.Val = 6
也可以把值類型封箱到接口類型中,只要它們實現(xiàn)這個接口即可。例如,假定MyStruct類型實現(xiàn)IMylnterface接口,如下所示:
interface IMylnterface {}
struct MyStruct : IMylnterface
{
public int Val;
}
接著把結(jié)構(gòu)封箱到一個IMylnterface類型中,如下所示:
MyStruct valTypel = new MyStruct ();
IMylnterface refType = valTypel;
然后使用一般的數(shù)據(jù)類型轉(zhuǎn)換語法對其拆箱:
MyStruct ValType2 = {MyStruct)refType;
從這些示例中可以看出,封箱是在沒有用戶干涉的情況下進行的(即不需要編寫任何代碼),但拆箱一個值需要進行顯式轉(zhuǎn)換,即需要進行數(shù)據(jù)類型轉(zhuǎn)換(封箱是隱式的,所以不需要進行數(shù)據(jù)類型轉(zhuǎn)換)。
讀者可能想知道為什么要這么做。封箱非常有用,有兩個非常重要的原因。首先,它允許在項的類型是object的集合(如ArrayList)中使用值類型。其次,有一個內(nèi)部機制允許在值類型(例如int和結(jié)構(gòu))上調(diào)用object方法。
最后需要注意的是,在訪問值類型內(nèi)容前,必須進行拆箱。
點擊加載更多評論>>