青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

山寨:不是最好的,是最適合我們的!歡迎體驗(yàn)山寨 中文版MSDN

Blog @ Blog

當(dāng)華美的葉片落盡,生命的脈絡(luò)才歷歷可見(jiàn)。 -- 聶魯達(dá)

常用鏈接

統(tǒng)計(jì)

積分與排名

BBS

Blog

Web

最新評(píng)論

C# 裝箱和拆箱[轉(zhuǎn)]

1、
      裝箱和拆箱是一個(gè)抽象的概念
2、
      裝箱是將值類(lèi)型轉(zhuǎn)換為引用類(lèi)型 ;拆箱是將引用類(lèi)型轉(zhuǎn)換為值類(lèi)型 
      利用裝箱和拆箱功能,可通過(guò)允許值類(lèi)型的任何值與Object 類(lèi)型的值相互轉(zhuǎn)換,將值類(lèi)型與引用類(lèi)型鏈接起來(lái)
例如:
int val = 100;
object obj = val;
Console.WriteLine (“對(duì)象的值 = {0}", obj);
這是一個(gè)裝箱的過(guò)程,是將值類(lèi)型轉(zhuǎn)換為引用類(lèi)型的過(guò)程

int val = 100;
object obj = val;
int num = (int) obj;
Console.WriteLine ("num: {0}", num);
這是一個(gè)拆箱的過(guò)程,是將值類(lèi)型轉(zhuǎn)換為引用類(lèi)型,再由引用類(lèi)型轉(zhuǎn)換為值類(lèi)型的過(guò)程

注:被裝過(guò)箱的對(duì)象才能被拆箱
3、
      .NET中,數(shù)據(jù)類(lèi)型劃分為值類(lèi)型引用(不等同于C++的指針)類(lèi)型,與此對(duì)應(yīng),內(nèi)存分配被分成了兩種方式,一為棧,二為堆,注意:是托管堆。
      值類(lèi)型只會(huì)在棧中分配。
      引用類(lèi)型分配內(nèi)存與托管堆。
      托管堆對(duì)應(yīng)于垃圾回收。

4:裝箱/拆箱是什么?
裝箱:用于在垃圾回收堆中存儲(chǔ)值類(lèi)型。裝箱是值類(lèi)型到 object 類(lèi)型或到此值類(lèi)型所實(shí)現(xiàn)的任何接口類(lèi)型的隱式轉(zhuǎn)換。
拆箱:從 object 類(lèi)型到值類(lèi)型或從接口類(lèi)型到實(shí)現(xiàn)該接口的值類(lèi)型的顯式轉(zhuǎn)換。

5:為何需要裝箱?(為何要將值類(lèi)型轉(zhuǎn)為引用類(lèi)型?)
一種最普通的場(chǎng)景是,調(diào)用一個(gè)含類(lèi)型為Object的參數(shù)的方法,該Object可支持任意為型,以便通用。當(dāng)你需要將一個(gè)值類(lèi)型(如Int32)傳入時(shí),需要裝箱。
另一種用法是,一個(gè)非泛型的容器,同樣是為了保證通用,而將元素類(lèi)型定義為Object。于是,要將值類(lèi)型數(shù)據(jù)加入容器時(shí),需要裝箱。

6:裝箱/拆箱的內(nèi)部操作。
裝箱:
對(duì)值類(lèi)型在堆中分配一個(gè)對(duì)象實(shí)例,并將該值復(fù)制到新的對(duì)象中。按三步進(jìn)行。
第一步:新分配托管堆內(nèi)存(大小為值類(lèi)型實(shí)例大小加上一個(gè)方法表指針和一個(gè)SyncBlockIndex)。
第二步:將值類(lèi)型的實(shí)例字段拷貝到新分配的內(nèi)存中。
第三步:返回托管堆中新分配對(duì)象的地址。這個(gè)地址就是一個(gè)指向?qū)ο蟮囊昧恕?
有人這樣理解:如果將Int32裝箱,返回的地址,指向的就是一個(gè)Int32。我認(rèn)為也不是不能這樣理解,但這確實(shí)又有問(wèn)題,一來(lái)它不全面,二來(lái)指向Int32并沒(méi)說(shuō)出它的實(shí)質(zhì)(在托管堆中)。
拆箱:
檢查對(duì)象實(shí)例,確保它是給定值類(lèi)型的一個(gè)裝箱值。將該值從實(shí)例復(fù)制到值類(lèi)型變量中。
有書(shū)上講,拆箱只是獲取引用對(duì)象中指向值類(lèi)型部分的指針,而內(nèi)容拷貝則是賦值語(yǔ)句之觸發(fā)。我覺(jué)得這并不要緊。最關(guān)鍵的是檢查對(duì)象實(shí)例的本質(zhì),拆箱和裝箱的類(lèi)型必需匹配,這一點(diǎn)上,在IL層上,看不出原理何在,我的猜測(cè),或許是調(diào)用了類(lèi)似GetType之類(lèi)的方法來(lái)取出類(lèi)型進(jìn)行匹配(因?yàn)樾枰獓?yán)格匹配)。

7:裝箱/拆箱對(duì)執(zhí)行效率的影響
顯然,從原理上可以看出,裝箱時(shí),生成的是全新的引用對(duì)象,這會(huì)有時(shí)間損耗,也就是造成效率降低。
那該如何做呢?
首先,應(yīng)該盡量避免裝箱。
比如上例2的兩種情況,都可以避免,在第一種情況下,可以通過(guò)重載函數(shù)來(lái)避免。第二種情況,則可以通過(guò)泛型來(lái)避免。
當(dāng)然,凡事并不能絕對(duì),假設(shè)你想改造的代碼為第三方程序集,你無(wú)法更改,那你只能是裝箱了。
對(duì)于裝箱/拆箱代碼的優(yōu)化,由于C#中對(duì)裝箱和拆箱都是隱式的,所以,根本的方法是對(duì)代碼進(jìn)行分析,而分析最直接的方式是了解原理結(jié)何查看反編譯的IL代碼。比如:在循環(huán)體中可能存在多余的裝箱,你可以簡(jiǎn)單采用提前裝箱方式進(jìn)行優(yōu)化。

8:對(duì)裝箱/拆箱更進(jìn)一步的了解
裝箱/拆箱并不如上面所講那么簡(jiǎn)單明了,比如:裝箱時(shí),變?yōu)橐脤?duì)象,會(huì)多出一個(gè)方法表指針,這會(huì)有何用處呢?
我們可以通過(guò)示例來(lái)進(jìn)一步探討。
舉個(gè)例子。
Struct A : ICloneable
{
public Int32 x;
public override String ToString() {
return String.Format(”{0}”,x);
}
public object Clone() {
return MemberwiseClone();
}
}
static void main()
{
A a;
a.x = 100;
Console.WriteLine(a.ToString());
Console.WriteLine(a.GetType());
A a2 = (A)a.Clone();
ICloneable c = a2;
Ojbect o = c.Clone();
}
5.0:a.ToString()。編譯器發(fā)現(xiàn)A重寫(xiě)了ToString方法,會(huì)直接調(diào)用ToString的指令。因?yàn)锳是值類(lèi)型,編譯器不會(huì)出現(xiàn)多態(tài)行為。因此,直接調(diào)用,不裝箱。(注:ToString是A的基類(lèi)System.ValueType的方法)
5.1:a.GetType(),GetType是繼承于System.ValueType的方法,要調(diào)用它,需要一個(gè)方法表指針,于是a將被裝箱,從而生成方法表指針,調(diào)用基類(lèi)的System.ValueType。(補(bǔ)一句,所有的值類(lèi)型都是繼承于System.ValueType的)。
5.2:a.Clone(),因?yàn)锳實(shí)現(xiàn)了Clone方法,所以無(wú)需裝箱。
5.3:ICloneable轉(zhuǎn)型:當(dāng)a2為轉(zhuǎn)為接口類(lèi)型時(shí),必須裝箱,因?yàn)榻涌谑且环N引用類(lèi)型。
5.4:c.Clone()。無(wú)需裝箱,在托管堆中對(duì)上一步已裝箱的對(duì)象進(jìn)行調(diào)用。
附:其實(shí)上面的基于一個(gè)根本的原理,因?yàn)槲囱b箱的值類(lèi)型沒(méi)有方法表指針,所以,不能通過(guò)值類(lèi)型來(lái)調(diào)用其上繼承的虛方法。另外,接口類(lèi)型是一個(gè)引用類(lèi)型。對(duì)此,我的理解,該方法表指針類(lèi)似C++的虛函數(shù)表指針,它是用來(lái)實(shí)現(xiàn)引用對(duì)象的多態(tài)機(jī)制的重要依據(jù)。

9:如何更改已裝箱的對(duì)象
對(duì)于已裝箱的對(duì)象,因?yàn)闊o(wú)法直接調(diào)用其指定方法,所以必須先拆箱,再調(diào)用方法,但再次拆箱,會(huì)生成新的棧實(shí)例,而無(wú)法修改裝箱對(duì)象。有點(diǎn)暈吧,感覺(jué)在說(shuō)繞口令。還是舉個(gè)例子來(lái)說(shuō):(在上例中追加change方法)
public void Change(Int32 x) {
this.x = x;
}
調(diào)用:
A a = new A();
a.x = 100;
Object o = a; //裝箱成o,下面,想改變o的值。
((A)o).Change(200); //改掉了嗎?沒(méi)改掉。
沒(méi)改掉的原因是o在拆箱時(shí),生成的是臨時(shí)的棧實(shí)例A,所以,改動(dòng)是基于臨時(shí)A的,并未改到裝箱對(duì)象。
(附:在托管C++中,允許直接取加拆箱時(shí)第一步得到的實(shí)例引用,而直接更改,但C#不行。)
那該如何是好?
嗯,通過(guò)接口方式,可以達(dá)到相同的效果。
實(shí)現(xiàn)如下:
interface IChange {
void Change(Int32 x);
}
struct A : IChange {

}
調(diào)用:
((IChange)o).Change(200);//改掉了嗎?改掉了。
為啥現(xiàn)在可以改?
在將o轉(zhuǎn)型為IChange時(shí),這里不會(huì)進(jìn)行再次裝箱,當(dāng)然更不會(huì)拆箱,因?yàn)閛已經(jīng)是引用類(lèi)型,再因?yàn)樗荌Change類(lèi)型,所以可以直接調(diào)用Change,于是,更改的也就是已裝箱對(duì)象中的字段了,達(dá)到期望的效果。

10、--------------------------
      將值類(lèi)型轉(zhuǎn)換為引用類(lèi)型,需要進(jìn)行裝箱操作(boxing):

1、首先從托管堆中為新生成的引用對(duì)象分配內(nèi)存。

2、然后將值類(lèi)型的數(shù)據(jù)拷貝到剛剛分配的內(nèi)存中。

3、返回托管堆中新分配對(duì)象的地址。

可以看出,進(jìn)行一次裝箱要進(jìn)行分配內(nèi)存和拷貝數(shù)據(jù)這兩項(xiàng)比較影響性能的操作。

將引用內(nèi)型轉(zhuǎn)換為值內(nèi)型,需要進(jìn)行拆箱操作(unboxing):

1、首先獲取托管堆中屬于值類(lèi)型那部分字段的地址,這一步是嚴(yán)格意義上的拆箱。

2、將引用對(duì)象中的值拷貝到位于線(xiàn)程堆棧上的值類(lèi)型實(shí)例中。

經(jīng)過(guò)這2步,可以認(rèn)為是同boxing是互反操作。嚴(yán)格意義上的拆箱,并不影響性能,但伴隨這之后的拷貝數(shù)據(jù)的操作就會(huì)同boxing操作中一樣影響性能。

11、-------------------------
NET的所有類(lèi)型都是由基類(lèi)System.Object繼承過(guò)來(lái)的,包括最常用的基礎(chǔ)類(lèi)型:int, byte, short,bool等等,就是說(shuō)所有的事物都是對(duì)象。如果申明這些類(lèi)型得時(shí)候都在堆(HEAP)中分配內(nèi)存,會(huì)造成極低的效率!(個(gè)中原因以及關(guān)于堆和棧得區(qū)別會(huì)在另一篇里單獨(dú)得說(shuō)說(shuō)!)
.NET如何解決這個(gè)問(wèn)題得了?正是通過(guò)將類(lèi)型分成值型(value)和引用型(regerencetype),C#中定義的值類(lèi)型包括原類(lèi)型(Sbyte、Byte、Short、Ushort、Int、Uint、Long、Ulong、Char、Float、Double、Bool、Decimal)、枚舉(enum)、結(jié)構(gòu)(struct),引用類(lèi)型包括:類(lèi)、數(shù)組、接口、委托、字符串等。
值型就是在棧中分配內(nèi)存,在申明的同時(shí)就初始化,以確保數(shù)據(jù)不為NULL;
引用型是在堆中分配內(nèi)存,初始化為null,引用型是需要GARBAGE COLLECTION來(lái)回收內(nèi)存的,值型不用,超出了作用范圍,系統(tǒng)就會(huì)自動(dòng)釋放!
下面就來(lái)說(shuō)裝箱和拆箱的定義!
裝箱就是隱式的將一個(gè)值型轉(zhuǎn)換為引用型對(duì)象。比如:
int i=0;
Syste.Object obj=i;

這個(gè)過(guò)程就是裝箱!就是將i裝箱!
拆箱就是將一個(gè)引用型對(duì)象轉(zhuǎn)換成任意值型!比如:
int i=0;
System.Object obj=i;
int j=(int)obj;

這個(gè)過(guò)程前2句是將i裝箱,后一句是將obj拆箱!

posted on 2009-09-27 14:45 isabc 閱讀(642) 評(píng)論(0)  編輯 收藏 引用


只有注冊(cè)用戶(hù)登錄后才能發(fā)表評(píng)論。
網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問(wèn)   Chat2DB   管理


廣告信息(免費(fèi)廣告聯(lián)系)

中文版MSDN:
歡迎體驗(yàn)

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            国产视频在线观看一区二区三区| 国产亚洲精品久| 欧美日产在线观看| 欧美日韩国产成人在线91| 欧美久久久久久久| 欧美视频亚洲视频| 国产精品自拍小视频| 国产午夜精品一区理论片飘花| 国产综合色产| 最新成人av在线| 一区二区精品国产| 欧美亚洲一区二区在线观看| 久久福利影视| 免费国产自线拍一欧美视频| 亚洲国产成人av在线| 亚洲免费电影在线观看| 亚洲一级片在线观看| 欧美制服丝袜| 欧美高清视频在线播放| 欧美色图首页| 国产一区在线视频| 日韩一级黄色片| 欧美一区午夜视频在线观看| 久久综合五月| 日韩视频一区二区| 校园春色国产精品| 男人的天堂成人在线| 国产精品99免费看| 韩国一区二区三区美女美女秀| 亚洲欧洲日韩女同| 午夜在线a亚洲v天堂网2018| 久久一综合视频| 日韩一级大片在线| 久久精品五月婷婷| 欧美视频一区二区三区| 狠狠久久综合婷婷不卡| 一区二区毛片| 六月婷婷一区| 正在播放欧美视频| 美女999久久久精品视频| 国产精品超碰97尤物18| 在线精品一区| 亚洲欧美日韩视频一区| 欧美激情第三页| 亚洲欧美日韩成人高清在线一区| 免费国产一区二区| 国产毛片精品视频| 99国内精品久久| 久久综合久久88| 亚洲午夜精品久久| 欧美电影美腿模特1979在线看| 国产美女精品视频免费观看| 亚洲日本欧美日韩高观看| 久久本道综合色狠狠五月| 亚洲人成小说网站色在线| 欧美在线看片| 国产精品看片资源| 日韩视频一区二区三区在线播放| 久久久久久国产精品一区| 日韩天堂在线观看| 免费看亚洲片| 国语自产精品视频在线看一大j8 | 亚洲国产日韩综合一区| 开元免费观看欧美电视剧网站| 久久久国产精品亚洲一区 | 亚洲国产精品成人| 欧美一区二区在线看| 欧美视频亚洲视频| 日韩亚洲在线观看| 欧美高清在线观看| 久久九九热免费视频| 国产欧美一区二区三区在线看蜜臀| 亚洲免费观看高清在线观看 | 亚洲成人在线网站| 久久不射电影网| 亚洲一区二区在线| 欧美三级小说| 一区二区三区视频观看| 亚洲欧洲综合| 欧美激情精品久久久六区热门| 一区二区三区在线高清| 久久人体大胆视频| 久久er99精品| 一区二区三区在线视频免费观看| 久久久国产精品一区| 欧美一站二站| 韩国三级在线一区| 久久人人超碰| 久久久视频精品| 永久555www成人免费| 美女久久一区| 看欧美日韩国产| 亚洲欧洲三级| 亚洲国产毛片完整版| 欧美国产一区二区在线观看| 91久久一区二区| 91久久一区二区| 欧美日韩高清免费| 亚洲自拍高清| 午夜精品久久久久久久蜜桃app | 在线视频精品一| 欧美激情一区二区三区全黄| 久久亚洲精品视频| 亚洲国产99| 亚洲国产第一| 欧美日韩在线精品一区二区三区| 在线视频欧美一区| 亚洲一区精品视频| 国产亚洲欧美激情| 美女尤物久久精品| 欧美成人嫩草网站| 亚洲一区二区三区乱码aⅴ蜜桃女 亚洲一区二区三区乱码aⅴ | 亚洲天堂久久| 亚洲字幕一区二区| 韩国欧美一区| 亚洲高清视频中文字幕| 欧美日韩亚洲视频| 欧美在线播放高清精品| 久久精品日产第一区二区三区| 亚洲福利一区| 一本大道久久a久久综合婷婷| 国产精品日韩精品欧美精品| 久久久福利视频| 能在线观看的日韩av| 亚洲一区二区三区成人在线视频精品 | 亚洲特级片在线| 国内视频一区| 亚洲国产一区二区a毛片| 国产精品久久9| 麻豆久久婷婷| 欧美特黄一区| 久久亚裔精品欧美| 欧美日韩国产成人在线91| 欧美永久精品| 欧美激情视频一区二区三区免费 | 99精品免费| 欧美一区二区三区免费观看视频 | 美女精品在线观看| 亚洲欧美第一页| 久久综合伊人77777麻豆| 亚洲永久字幕| 美国三级日本三级久久99| 亚洲欧美另类国产| 麻豆精品在线视频| 欧美在线免费观看亚洲| 欧美激情一区二区三区四区| 久久爱www.| 欧美日韩国语| 噜噜噜躁狠狠躁狠狠精品视频| 欧美日韩另类综合| 欧美α欧美αv大片| 国产精品伦理| 亚洲激情av| 韩国成人理伦片免费播放| 99精品国产在热久久| 精品动漫3d一区二区三区免费版| 一区二区三区四区五区精品视频 | 香港久久久电影| 欧美激情第五页| 久久只有精品| 国产精品一区二区视频| 亚洲精品日韩在线| 在线欧美日韩精品| 亚洲女同精品视频| 一区二区三区波多野结衣在线观看| 欧美在线亚洲在线| 午夜在线播放视频欧美| 欧美日韩国产区一| 亚洲大胆在线| 激情久久五月| 欧美一级片在线播放| 亚洲一区www| 欧美日韩和欧美的一区二区| 欧美激情欧美激情在线五月| 国产一区香蕉久久| 亚洲欧美日韩一区| 亚洲一区二区三区午夜| 欧美精品一区三区| 久久爱www久久做| 欧美高清成人| 国产在线拍揄自揄视频不卡99| 日韩一区二区电影网| 亚洲精品欧美日韩| 蜜臀久久99精品久久久久久9 | 国产午夜精品全部视频在线播放| 一本色道**综合亚洲精品蜜桃冫 | 国产午夜精品麻豆| 亚洲一区在线直播| 亚洲欧美日韩精品| 欧美日韩一区二区三区免费| 亚洲激情一区二区| 亚洲欧洲视频在线| 欧美成ee人免费视频| 欧美国产精品日韩| 亚洲国产日本| 欧美电影在线播放| 亚洲国产清纯| 亚洲精品人人| 欧美日本免费| 一本久久综合亚洲鲁鲁|