您的位置:知识库 »

C# 装箱和拆箱[整理]

作者: 桦林  来源: 博客园  发布时间: 2008-09-08 18:06  阅读: 9525 次  推荐: 1   原文链接   [收藏]  
[1] C# 装箱和拆箱[整理]
[2] C# 装箱和拆箱[整理]
7. 装箱/拆箱对执行效率的影响
    显然,从原理上可以看出,装箱时,生成的是全新的引用对象,这会有时间损耗,也就是造成效率降低。 
    那该如何做呢?
    首先,应该尽量避免装箱。
    比如上例2的两种情况,都可以避免,在第一种情况下,可以通过重载函数来避免。第二种情况,则可以通过泛型来避免。
    当然,凡事并不能绝对,假设你想改造的代码为第三方程序集,你无法更改,那你只能是装箱了。
    对于装箱/拆箱代码的优化,由于C#中对装箱和拆箱都是隐式的,所以,根本的方法是对代码进行分析,而分析最直接的方式是了解原理结何查看反编译的IL代码。比如:在循环体中可能存在多余的装箱,你可以简单采用提前装箱方式进行优化。

 

8. 对装箱/拆箱更进一步的了解
    装箱/拆箱并不如上面所讲那么简单明了,比如:装箱时,变为引用对象,会多出一个方法表指针,这会有何用处呢?
    我们可以通过示例来进一步探讨。

 

Code
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,于是,更改的也就是已装箱对象中的字段了,达到期望的效果。

 

10. 将值类型转换为引用类型,需要进行装箱操作(boxing):

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

      2)然后将值类型的数据拷贝到刚刚分配的内存中。

      3)返回托管堆中新分配对象的地址。

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

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

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

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

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

 

 

[第1页][第2页]
1
1
标签:C# 装箱 拆箱

热门文章

    最新文章

      最新新闻

        热门新闻