装箱与拆箱

一、数组

      装箱和拆箱是一个抽象的概念 ide

二、函数

      装箱是将值类型转换为引用类型 ;拆箱是将引用类型转换为值类型 性能

      利用装箱和拆箱功能,可经过容许值类型的任何值与Object 类型的值相互转换,将值类型与引用类型连接起来 优化

例如: this

int val = 100; 线程

object obj = val; 指针

Console.WriteLine (“对象的值 = {0}", obj); orm

这是一个装箱的过程,是将值类型转换为引用类型的过程 对象


int val = 100; 

object obj = val; 

int num = (int) obj; 

Console.WriteLine ("num: {0}", num); 

这是一个拆箱的过程,是将值类型转换为引用类型,再由引用类型转换为值类型的过程 


注:被装过箱的对象才能被拆箱

三、

      .NET中,数据类型划分为值类型和引用(不等同于C++的指针)类型,与此对应,内存分配被分红了两种方式,一为栈,二为堆,注意:是托管堆。

      值类型只会在栈中分配。

      引用类型分配内存与托管堆。

      托管堆对应于垃圾回收。



4:装箱/拆箱是什么? 

装箱:用于在垃圾回收堆中存储值类型。装箱是值类型到 object 类型或到此值类型所实现的任何接口类型的隐式转换。 

拆箱:从 object 类型到值类型或从接口类型到实现该接口的值类型的显式转换。 


5:为什么须要装箱?(为什么要将值类型转为引用类型?) 

一种最普通的场景是,调用一个含类型为Object的参数的方法,该Object可支持任意为型,以便通用。当你须要将一个值类型(如Int32)传入时,须要装箱。 

另外一种用法是,一个非泛型的容器,一样是为了保证通用,而将元素类型定义为Object。因而,要将值类型数据加入容器时,须要装箱。 


6:装箱/拆箱的内部操做。 

装箱: 

对值类型在堆中分配一个对象实例,并将该值复制到新的对象中。按三步进行。 

第一步:新分配托管堆内存(大小为值类型实例大小加上一个方法表指针和一个SyncBlockIndex)。 

第二步:将值类型的实例字段拷贝到新分配的内存中。 

第三步:返回托管堆中新分配对象的地址。这个地址就是一个指向对象的引用了。 

有人这样理解:若是将Int32装箱,返回的地址,指向的就是一个Int32。我认为也不是不能这样理解,但这确实又有问题,一来它不全面,二来指向Int32并没说出它的实质(在托管堆中)。 

拆箱:

检查对象实例,确保它是给定值类型的一个装箱值。将该值从实例复制到值类型变量中。 

有书上讲,拆箱只是获取引用对象中指向值类型部分的指针,而内容拷贝则是赋值语句之触发。我以为这并没关系。最关键的是检查对象实例的本质,拆箱和装箱的类型必需匹配,这一点上,在IL层上,看不出原理何在,个人猜想,或许是调用了相似GetType之类的方法来取出类型进行匹配(由于须要严格匹配)。 


7:装箱/拆箱对执行效率的影响 

显然,从原理上能够看出,装箱时,生成的是全新的引用对象,这会有时间损耗,也就是形成效率下降。 

那该如何作呢? 

首先,应该尽可能避免装箱。 

好比上例2的两种状况,均可以免,在第一种状况下,能够经过重载函数来避免。第二种状况,则能够经过泛型来避免。 

固然,凡事并不能绝对,假设你想改造的代码为第三方程序集,你没法更改,那你只能是装箱了。 

对于装箱/拆箱代码的优化,因为C#中对装箱和拆箱都是隐式的,因此,根本的方法是对代码进行分析,而分析最直接的方式是了解原理结何查看反编译的IL代码。好比:在循环体中可能存在多余的装箱,你能够简单采用提早装箱方式进行优化。 


8:对装箱/拆箱更进一步的了解 

装箱/拆箱并不如上面所讲那么简单明了,好比:装箱时,变为引用对象,会多出一个方法表指针,这会有何用处呢? 

咱们能够经过示例来进一步探讨。 

举个例子。 

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()。编译器发现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:ICloneable转型:当a2为转为接口类型时,必须装箱,由于接口是一种引用类型。 

5.4:c.Clone()。无需装箱,在托管堆中对上一步已装箱的对象进行调用。 

附:其实上面的基于一个根本的原理,由于未装箱的值类型没有方法表指针,因此,不能经过值类型来调用其上继承的虚方法。另外,接口类型是一个引用类型。对此,个人理解,该方法表指针相似C++的虚函数表指针,它是用来实现引用对象的多态机制的重要依据。 


9:如何更改已装箱的对象 

对于已装箱的对象,由于没法直接调用其指定方法,因此必须先拆箱,再调用方法,但再次拆箱,会生成新的栈实例,而没法修改装箱对象。有点晕吧,感受在说绕口令。仍是举个例子来讲:(在上例中追加change方法) 

public void Change(Int32 x) { 

this.x = x; 

调用: 

A a = new A(); 

a.x = 100; 

Object o = a; //装箱成o,下面,想改变o的值。 

((A)o).Change(200); //改掉了吗?没改掉。 

没改掉的缘由是o在拆箱时,生成的是临时的栈实例A,因此,改动是基于临时A的,并未改到装箱对象。 

(附:在托管C++中,容许直接取加拆箱时第一步获得的实例引用,而直接更改,但C#不行。) 

那该如何是好? 

嗯,经过接口方式,能够达到相同的效果。 

实现以下: 

interface IChange { 

void Change(Int32 x); 

struct A : IChange { 

… 

调用: 

((IChange)o).Change(200);//改掉了吗?改掉了。 

为啥如今能够改? 

在将o转型为IChange时,这里不会进行再次装箱,固然更不会拆箱,由于o已是引用类型,再由于它是IChange类型,因此能够直接调用Change,因而,更改的也就是已装箱对象中的字段了,达到指望的效果。


十、--------------------------

      将值类型转换为引用类型,须要进行装箱操做(boxing):


一、首先从托管堆中为新生成的引用对象分配内存。


二、而后将值类型的数据拷贝到刚刚分配的内存中。


三、返回托管堆中新分配对象的地址。


能够看出,进行一次装箱要进行分配内存和拷贝数据这两项比较影响性能的操做。


将引用内型转换为值内型,须要进行拆箱操做(unboxing):


一、首先获取托管堆中属于值类型那部分字段的地址,这一步是严格意义上的拆箱。


二、将引用对象中的值拷贝到位于线程堆栈上的值类型实例中。


通过这2步,能够认为是同boxing是互反操做。严格意义上的拆箱,并不影响性能,但伴随这以后的拷贝数据的操做就会同boxing操做中同样影响性能。


十一、-------------------------

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,引用型是须要GARBAGE COLLECTION来回收内存的,值型不用,超出了做用范围,系统就会自动释放!

下面就来讲装箱和拆箱的定义!

装箱就是隐式的将一个值型转换为引用型对象。好比:

int i=0;

Syste.Object obj=i;

这个过程就是装箱!就是将i装箱!

拆箱就是将一个引用型对象转换成任意值型!好比:

int i=0;

System.Object obj=i;

int j=(int)obj;

这个过程前2句是将i装箱,后一句是将obj拆箱!

相关文章
相关标签/搜索