版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
1、关于C#装箱与拆箱的理解NET重要技术和基础之一的CTS(Common Type System。顾名思义,CTS就是为了实现在应用程序声明和使用这些类型时必须遵循的规则而存在的通用类型系统。.Net将整个系统的类型分成两大类 Value Type 和 Reference Type。,多数的OO语言存在这个弱点,原因就是因为他们的原类型没有共同的基点,于是他们在本质上并不是真正的对象C+更依赖于对象,而非面向对象。.Net环境的CTS 给我们带来了方便。第一、CTS中的所有东西都是对象;第二、所有的对象都源自一个基类System.Object类型。这就是所谓的单根层次结构(singly root
2、ed hierarchy)关于System.Object的详细资料请参考微软的技术文档。CTS Value Type的一个最大的特点是它们不能为null,Value Type的变量总有一个值。在传递Value Type的变量时,实际传递的是变量的值,而非底层对象的“引用”。CTS Reference Type就好像是类型安全的指针,它可以为null。当值为null时,说明没有引用或类型指向某个对象。声明一个引用类型的变量时,被操作的是此变量的引用(地址),而不是数据。 1、装箱和拆箱是一个抽象的概念 2、装箱是将值类型转换为引用类型 ;拆箱是将引用类型转换为值类型 利用装箱和拆箱功能,可通过允
3、许值类型的任何值与Object 类型的值相互转换,将值类型与引用类型链接起来 例如: int val = 100; object obj = val; Console.WriteLine (“对象的值 = 0", obj; 这是一个装箱的过程,是将值类型转换为引用类型的过程 int val = 100; object obj = val; int num = (int obj; Console.WriteLine ("num: 0", num; 这是一个拆箱的过程,是将值类型转换为引用类型,再由引用类型转换为值类型的过程 注:被装过箱的对象才能被拆箱3、.NET中
4、,数据类型划分为值类型和引用(不等同于C+的指针类型,与此对应,内存分配被分成了两种方式,一为栈,二为堆,注意:是托管堆。值类型只会在栈中分配。引用类型分配内存与托管堆。托管堆对应于垃圾回收。4:装箱/拆箱是什么? 装箱:用于在垃圾回收堆中存储值类型。装箱是值类型到 object 类型或到此值类型所实现的任何接口类型的隐式转换。 拆箱:从 object 类型到值类型或从接口类型到实现该接口的值类型的显式转换。5:为何需要装箱?(为何要将值类型转为引用类型? 一种最普通的场景是,调用一个含类型为Object的参数的方法,该Object可支持任意为型,以便通用。当你需要将一个值类型(如Int32传
5、入时,需要装箱。 另一种用法是,一个非泛型的容器,同样是为了保证通用,而将元素类型定义为Object。于是,要将值类型数据加入容器时,需要装箱。6:装箱/拆箱的内部操作。 装箱: 对值类型在堆中分配一个对象实例,并将该值复制到新的对象中。按三步进行。 第一步:新分配托管堆内存(大小为值类型实例大小加上一个方法表指针和一个SyncBlockIndex。 第二步:将值类型的实例字段拷贝到新分配的内存中。 第三步:返回托管堆中新分配对象的地址。这个地址就是一个指向对象的引用了。 有人这样理解:如果将Int32装箱,返回的地址,指向的就是一个Int32。我认为也不是不能这样理解,但这确实又有问题,一来
6、它不全面,二来指向Int32并没说出它的实质(在托管堆中。 拆箱:检查对象实例,确保它是给定值类型的一个装箱值。将该值从实例复制到值类型变量中。 有书上讲,拆箱只是获取引用对象中指向值类型部分的指针,而内容拷贝则是赋值语句之触发。我觉得这并不要紧。最关键的是检查对象实例的本质,拆箱和装箱的类型必需匹配,这一点上,在IL层上,看不出原理何在,我的猜测,或许是调用了类似GetType之类的方法来取出类型进行匹配(因为需要严格匹配。7:装箱/拆箱对执行效率的影响 显然,从原理上可以看出,装箱时,生成的是全新的引用对象,这会有时间损耗,也就是造成效率降低。 那该如何做呢? 首先,应该尽量避免装箱。 比
7、如上例2的两种情况,都可以避免,在第一种情况下,可以通过重载函数来避免。第二种情况,则可以通过泛型来避免。 当然,凡事并不能绝对,假设你想改造的代码为第三方程序集,你无法更改,那你只能是装箱了。 对于装箱/拆箱代码的优化,由于C#中对装箱和拆箱都是隐式的,所以,根本的方法是对代码进行分析,而分析最直接的方式是了解原理结何查看反编译的IL代码。比如:在循环体中可能存在多余的装箱,你可以简单采用提前装箱方式进行优化。8:对装箱/拆箱更进一步的了解 装箱/拆箱并不如上面所讲那么简单明了,比如:装箱时,变为引用对象,会多出一个方法表指针,这会有何用处呢? 我们可以通过示例来进一步探讨。 举个例子。 S
8、truct 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 = (Aa.Clone(; ICloneable c = a2; Ojbect o = c.Clone(; 5.
9、0:a.ToString(。编译器发现A重写了ToString方法,会直接调用ToString的指令。因为A是值类型,编译器不会出现多态行为。因此,直接调用,不装箱。(注:ToString是A的基类System.ValueType的方法 5.1:a.GetType(,GetType是继承于System.ValueType的方法,要调用它,需要一个方法表指针,于是a将被装箱,从而生成方法表指针,调用基类的System.ValueType。(补一句,所有的值类型都是继承于System.ValueType的。 5.2:a.Clone(,因为A实现了Clone方法,所以无需装箱。 5.3:IClone
10、able转型:当a2为转为接口类型时,必须装箱,因为接口是一种引用类型。 5.4:c.Clone(。无需装箱,在托管堆中对上一步已装箱的对象进行调用。 附:其实上面的基于一个根本的原理,因为未装箱的值类型没有方法表指针,所以,不能通过值类型来调用其上继承的虚方法。另外,接口类型是一个引用类型。对此,我的理解,该方法表指针类似C+的虚函数表指针,它是用来实现引用对象的多态机制的重要依据。9:如何更改已装箱的对象 对于已装箱的对象,因为无法直接调用其指定方法,所以必须先拆箱,再调用方法,但再次拆箱,会生成新的栈实例,而无法修改装箱对象。有点晕吧,感觉在说绕口令。还是举个例子来说:(在上例中追加ch
11、ange方法 public void Change(Int32 x this.x = x; 调用: A a = new A(; a.x = 100; Object o = a; /装箱成o,下面,想改变o的值。 (Ao.Change(200; /改掉了吗?没改掉。 没改掉的原因是o在拆箱时,生成的是临时的栈实例A,所以,改动是基于临时A的,并未改到装箱对象。 (附:在托管C+中,允许直接取加拆箱时第一步得到的实例引用,而直接更改,但C#不行。 那该如何是好? 嗯,通过接口方式,可以达到相同的效果。 实现如下: interface IChange void Change(Int32 x; str
12、uct A : IChange 调用: (IChangeo.Change(200;/改掉了吗?改掉了。 为啥现在可以改? 在将o转型为IChange时,这里不会进行再次装箱,当然更不会拆箱,因为o已经是引用类型,再因为它是IChange类型,所以可以直接调用Change,于是,更改的也就是已装箱对象中的字段了,达到期望的效果。10、将值类型转换为引用类型,需要进行装箱操作(boxing:1、首先从托管堆中为新生成的引用对象分配内存。2、然后将值类型的数据拷贝到刚刚分配的内存中。3、返回托管堆中新分配对象的地址。可以看出,进行一次装箱要进行分配内存和拷贝数据这两项比较影响性能的操作。将引用内型转
13、换为值内型,需要进行拆箱操作(unboxing:1、首先获取托管堆中属于值类型那部分字段的地址,这一步是严格意义上的拆箱。2、将引用对象中的值拷贝到位于线程堆栈上的值类型实例中。经过这2步,可以认为是同boxing是互反操作。严格意义上的拆箱,并不影响性能,但伴随这之后的拷贝数据的操作就会同boxing操作中一样影响性能。11、-NET的所有类型都是由基类System.Object继承过来的,包括最常用的基础类型:int, byte, short,bool等等,就是说所有的事物都是对象。如果申明这些类型得时候都在堆(HEAP中分配内存,会造成极低的效率!(个中原因以及关于堆和栈得区别会在另一篇里单独得说说!.NET如何解决这个问题得了?正是通过将类型分成值型(value和引用型(regerencetype,C#中定义的值类型包括原类型(Sbyte、Byte、Short、Ushort、Int、Uint、Long、Ulong、Char、Float、Double、Bool、Decimal)、枚举(enum、结构(struct,引用类型包括:类、数组、接口、委托、字符串等。值型就是在栈中分配内存,在申明的同时就初始化,以确保数据不为NULL;引用型是在堆中分配内存,初始化为null,引用型是需
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 二类疫苗采购流程制度
- 街道采购管理制度
- 采购项目信息管理制度
- 采购领用制度
- 采购验货入库制度
- 金融企业采购制度
- 铁总物资采购管理制度
- 第8章 实数相关计算必考三大类型90题(必考点分类集训)(教师版)-人教版(2024)七下
- 七年级下学期第一次月考易错题复习(13个考点36题)(教师版)-人教版(2024)七下
- 2026年设计资质合同(1篇)
- 会计毕业实习报告1000字(30篇)
- 北师大版六年级下册《正比例》课件市公开课一等奖省赛课获奖课件
- 颌面部骨折围手术期的护理
- 地铁行业沟通技巧分析
- 2023年六年级小升初自荐信简历
- 清明时节 奠说巴人获奖科研报告
- 主蒸汽管道更换施工方案
- 如何给领导拍照
- 初中校本课程-【校本课程】春节教学课件设计
- 注塑模具相关零件加工工艺过程卡片
- 急性上消化道出血中心建设PPT文档
评论
0/150
提交评论