深入理解C# 装箱和拆箱(整理篇)

装箱(boxing)和拆箱(unboxing)是C#类型系统的核心概念.是不同于C与C++的新概念!,通过装箱和拆箱操作,能够在值类型和引用类型中架起一做桥梁.换言之,可以轻松的实现值类型与引用类型的互相转换,装箱和拆箱能够统一考察系统,任何类型的值最终都可以按照对象进行处理.

装箱和拆箱是值类型和引用类型之间相互转换是要执行的操作。

1. 装箱在值类型向引用类型转换时发生

2. 拆箱在引用类型向值类型转换时发生

//1、
//   装箱和拆箱是一个抽象的概念
//2、
//   装箱是将值类型转换为引用类型 ;拆箱是将引用类型转换为值类型
//   利用装箱和拆箱功能,可通过允许值类型的任何值与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中,数据类型划分为值类型和引用(不等同于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,于是,更改的也就是已//装箱对象中的字段了,达到期望的效果。
//10、--------------------------
 //   将值类型转换为引用类型,需要进行装箱操作(boxing):
//1、首先从托管堆中为新生成的引用对象分配内存。
//2、然后将值类型的数据拷贝到刚刚分配的内存中。
//3、返回托管堆中新分配对象的地址。
//可以看出,进行一次装箱要进行分配内存和拷贝数据这两项比较影响性能的操作。
//将引用内型转换为值内型,需要进行拆箱操作(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,引用型是需要GARBAGE COLLECTION来//回收内存的,值型不用,超出了作用范围,系统就会自动释放!
//下面就来说装箱和拆箱的定义!
//装箱就是隐式的将一个值型转换为引用型对象。比如:
int i=0;
Syste.Object obj=i;
//这个过程就是装箱!就是将i装箱!
//拆箱就是将一个引用型对象转换成任意值型!比如:
int i=0;
System.Object obj=i;
int j=(int)obj;
//这个过程前2句是将i装箱,后一句是将obj拆箱!1、

总结

以上所述是小编给大家介绍的C# 装箱和拆箱(整理篇),希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对我们网站的支持!

(0)

相关推荐

  • 解析C#中的装箱与拆箱的详解

    装箱和拆箱是值类型和引用类型之间相互转换是要执行的操作. 1. 装箱在值类型向引用类型转换时发生2. 拆箱在引用类型向值类型转换时发生光上述两句话不难理解,但是往深处了解,就需要一些篇幅来解释了.我们先看装箱时都会发生什么事情,下面是一行最简单的装箱代码 复制代码 代码如下: object obj = 1; 这行语句将整型常量1赋给object类型的变量obj: 众所周知常量1是值类型,值类型是要放在栈上的,而object是引用类型,它需要放在堆上:要把值类型放在堆上就需要执行一次装箱操作.这行

  • 轻松学习C#的装箱与拆箱

    首先看一看什么是装箱和拆箱?        简单的来说:        装箱就是值类型转换为引用类型:        拆箱就是引用类型转换为值类型.        值类型,包括原类型(Sbyte.Byte.Short.Ushort.Int.Uint.Long.Ulong.Char.Float.Double.Bool.Decimal).枚举 (enum) .结构 (struct).        引用类型包括类.数组.接口.委托.字符串等. 装箱:值类型到引用类型或到此值类型所实现的任何接口类型的

  • 再议C#中的装箱与拆箱的问题详解

    上一篇写了一下装箱拆箱的定义和IL分析,这一篇我们看下使用泛型和不使用泛型引发装箱拆箱的情况1.使用非泛型集合时引发的装箱和拆箱操作 看下面的一段代码: 复制代码 代码如下: var array = new ArrayList();array.Add(1);array.Add(2); foreach (int value in array){Console.WriteLine("value is {0}",value);} 代码声明了一个ArrayList对象,向ArrayList中添

  • c#装箱和拆箱知识整理

    1.装箱和拆箱是一个抽象的概念 2.装箱是将值类型转换为引用类型 : 拆箱是将引用类型转换为值类型 利用装箱和拆箱功能,可通过允许值类型的任何值与Object 类型的值相互转换,将值类型与引用类型链接起来 例如: 复制代码 代码如下: int val = 100; object obj = val; Console.WriteLine ("对象的值 = {0}", obj); 这是一个装箱的过程,是将值类型转换为引用类型的过程 复制代码 代码如下: int val = 100; obj

  • C#实现装箱与拆箱操作简单实例

    本文以一个简单实例讲述了C#装箱和拆箱操作的实现方法,简单来说装箱是将值类型转换为引用类型:拆箱是将引用类型转换为值类型,是涉及栈和堆的使用方面的东西,学过C#的人应该都知道,所以这里就不哆嗦了,本例代码也是面向C#新手的,非常简单. 具体实现代码如下: using System; using System.Collections.Generic; using System.Linq; using System.Text; namespace UnBoxing { class Program {

  • C#装箱和拆箱操作实例分析

    本文实例讲述了C#装箱和拆箱操作.分享给大家供大家参考,具体如下: 1. C#中的装箱 C#中的装箱就是把一个值类型隐式地转换为object类型,转换过程中采用的是值的拷贝而不是引用,这从下面的例子可以看出: using System; public class Test { public static void Main(String[] args) { int i = 10; //将值类型的i装箱 //需要注意的是:这里的装箱采用的是值的拷贝 object obj = i; //检验是否装箱

  • C#装箱和拆箱原理详解

    .NET包含一个特殊的Object类,可以接受任意的数据类型的值,当所传递或所赋值的类型不是一个特定的数据类型时,object类就提供了一种传递参数和赋值的通用方法.赋给object的值必须作为引用类型,并存放砸托管堆中. 装箱: int age = 24; object refAge= age; 可以看的出,第一条语句创建一个变量age,并将值放在托管栈中: 第二条语句将age的值赋给引用类型.它将值24放在托管堆中. 这个值类型包装为引用类型的过程,称为装箱. 拆箱: 相反,将引用类型转换为

  • 深入理解C# 装箱和拆箱(整理篇)

    装箱(boxing)和拆箱(unboxing)是C#类型系统的核心概念.是不同于C与C++的新概念!,通过装箱和拆箱操作,能够在值类型和引用类型中架起一做桥梁.换言之,可以轻松的实现值类型与引用类型的互相转换,装箱和拆箱能够统一考察系统,任何类型的值最终都可以按照对象进行处理. 装箱和拆箱是值类型和引用类型之间相互转换是要执行的操作. 1. 装箱在值类型向引用类型转换时发生 2. 拆箱在引用类型向值类型转换时发生 //1. // 装箱和拆箱是一个抽象的概念 //2. // 装箱是将值类型转换为引

  • Java中的装箱和拆箱深入理解

    自动装箱和拆箱问题是Java中一个老生常谈的问题了,今天我们就来一些看一下装箱和拆箱中的若干问题.本文先讲述装箱和拆箱最基本的东西,再来看一下面试笔试中经常遇到的与装箱.拆箱相关的问题. 一.什么是装箱?什么是拆箱? 在前面的文章中提到,Java为每种基本数据类型都提供了对应的包装器类型,至于为什么会为每种基本数据类型提供包装器类型在此不进行阐述,有兴趣的朋友可以查阅相关资料.在Java SE5之前,如果要生成一个数值为10的Integer对象,必须这样进行: 复制代码 代码如下: Intege

  • 关于C#理解装箱与拆箱

    目录 1.理解装箱 2.理解拆箱 3.生成的 IL 代码 4.实际应用 5.小结 1.理解装箱 简单地说,装箱就是将一个值类型的数据存储在一个引用类型的变量中. 假设你一个方法中创建了一个 int 类型的本地变量,你要将这个值类型表示为一个引用类型,那么就表示你对这个值进行了装箱操作,如下所示: static void SimpleBox() { int myInt = 25; // 装箱操作 object boxedInt = myInt; } 确切地说,装箱的过程就是将一个值类型分配给 Ob

  • 深入理解Java中的装箱和拆箱

    前言 自动装箱和拆箱问题是Java中一个老生常谈的问题了,今天我们就来一些看一下装箱和拆箱中的若干问题.本文先讲述装箱和拆箱最基本的东西,再来看一下面试笔试中经常遇到的与装箱.拆箱相关的问题. 若有不正之处,请谅解和批评指正,不胜感激. 一.什么是装箱?什么是拆箱? 在前面的文章中提到,Java为每种基本数据类型都提供了对应的包装器类型,至于为什么会为每种基本数据类型提供包装器类型在此不进行阐述,有兴趣的朋友可以查阅相关资料.在Java SE5之前,如果要生成一个数值为10的Integer对象,

  • Java 装箱与拆箱详解及实例代码

    Java 装箱与拆箱详解 前言: 要理解装箱和拆箱的概念,就要理解Java数据类型 装箱:把基本类型用它们相应的引用类型包装起来,使其具有对象的性质.int包装成Integer.float包装成Float 拆箱:和装箱相反,将引用类型的对象简化成值类型的数据 Integer a = 100; 这是自动装箱 (编译器调用的是static Integer valueOf(int i)) int b = new Integer(100); 这是自动拆箱 看下面一段代码 m1 public class

  • 基于数据类型转换(装箱与拆箱)与常量详解

    隐式转换[自动类型转换]: 两种类型要兼容,原类型值域要小于目标类型值域,可以简单的理解为由小转大. public class Test { private void Start() { int a = 10; float b = a;//int 类型隐式转换为 float 类型 } } 显示转换[强制类型转换]: 两种类型要兼容,原类型值域要大于目标类型值域,可以简单的理解为由大转小. [缺点]:1.数据溢出.2.精度丢失. 数值类型之间的转换. public class Test { pri

  • C#装箱和拆箱的原理介绍

    我们知道,值类型的变量是在堆栈上分配内存的,而引用类型包括System.Object的对象是在堆上分配内存的,基于这一特点,当值类型被类型转换时,会在堆栈和堆上进行一系列的操作,这就是装箱和拆箱的来源.充分理解装箱和拆箱,有助于程序员编写高效率的代码. 1.装箱和拆箱的基本概念 我们知道,所有的值类型都继承自System.ValueType,而System.ValueType继承自System.Object.所有的值类型对象都分配在堆栈上,而所有的引用类型包括System.Object对象都分配

  • Java详细分析讲解自动装箱自动拆箱与Integer缓存的使用

    目录 1. 前言 2. 包装类 3. 自动装箱与自动拆箱 4. Interger缓存 5. 回答题目 1. 前言 自动装箱和自动拆箱是什么?Integer缓存是什么?它们之间有什么关系? 先来看一道题目. Integer a = new Integer(1); Integer b = new Integer(1); System.out.println(a==b); Integer c = 1; Integer d = 1; System.out.println(c==d); Integer e

随机推荐