C#中的装箱和拆箱是值类型与引用类型之间相互转换时产生的操作,是理解C#类型系统的基础知识点,直接关系到代码的运行性能。

什么是装箱
装箱是值类型转换为object类型或者该值类型所实现的任何接口类型的过程。当执行装箱操作时,CLR会在托管堆上分配一块内存,将值类型的数据复制到这块内存中,然后返回这块内存的引用,这个引用就是一个object类型的对象。
下面是一段装箱的示例代码:
// 定义值类型变量 int num = 10; // 执行装箱操作,将int值类型转换为object引用类型 object obj = num; // 也可以装箱到接口类型,假设int实现了IComparable接口 System.IComparable comparable = num;
什么是拆箱
拆箱是从object类型或者接口类型显式转换回对应的值类型的过程。拆箱操作分为两步:首先,检查object实例是否包含给定值类型的有效数据,也就是检查这个object是不是对应值类型的装箱结果;然后,如果检查通过,就将存储在托管堆中的值复制到值类型的变量中。
拆箱的示例代码如下:
object obj = 10; // 执行拆箱操作,必须显式指定要转换的值类型 int num = (int)obj; // 拆箱到接口类型对应的实现类型 System.IComparable comparable = 20; int num2 = (int)comparable;
装箱和拆箱的核心区别
1. 转换方向不同
装箱是值类型到引用类型的转换,方向是值类型 -> object/接口类型;拆箱是引用类型到值类型的转换,方向是object/接口类型 -> 值类型,二者是相反的操作。
2. 性能开销不同
装箱操作需要在托管堆上分配内存,还需要进行数据复制,会产生一定的性能开销;拆箱操作虽然不需要分配新的内存,但需要类型检查,同时也会进行数据复制,同样有性能消耗。频繁的装箱拆箱操作会导致程序性能下降,尤其是在循环等高频执行的代码块中。
3. 操作要求不同
装箱操作可以隐式完成,不需要开发者显式指定转换类型;而拆箱操作必须显式指定要转换的目标值类型,并且拆箱的类型必须和装箱时的原始值类型完全一致,否则会抛出InvalidCastException异常。
4. 内存分配位置不同
装箱后的对象存储在托管堆上,由垃圾回收器管理生命周期;而原始的值类型如果是在方法内部定义的,原本是存储在栈上的,装箱后数据被复制到堆中,栈上的原始值类型和堆上的装箱对象是相互独立的。
装箱拆箱的注意事项
为了避免不必要的性能损耗,在开发中应当尽量减少装箱拆箱操作。比如可以使用泛型集合List<int>代替非泛型的ArrayList,因为ArrayList存储值类型时会发生装箱,而泛型集合可以直接存储值类型,避免装箱操作。
下面的代码展示了不必要的装箱和优化的写法:
// 不推荐的写法,会产生装箱
System.Collections.ArrayList arrayList = new System.Collections.ArrayList();
for (int i = 0; i < 1000; i++)
{
arrayList.Add(i); // int装箱为object
}
// 推荐的写法,无装箱操作
System.Collections.Generic.List<int> list = new System.Collections.Generic.List<int>();
for (int i = 0; i < 1000; i++)
{
list.Add(i); // 直接存储int值类型
}
另外,在拆箱操作时一定要确保目标类型正确,否则会引发运行时异常,可以在拆箱前先使用is关键字进行类型检查,避免异常抛出。