C++:最强大的.NET语言之装箱
发布时间:2006-05-30 15:37:36 来源:天极开发 网友评论 0 条 再论类型
在讨论装箱(boxing)之前,有必要弄清楚为什么值类型与引用类型之间会有所区别。
一个含有数值的值类型的实例,和一个指向对象的引用类型的实例,它们有什么区别呢?除了存储对象所需的内存之外,每一个对象都会有一个对象头,目的是为面向对象的编程提供基本的服务,如存在虚方法的类,嵌入其中的元数据等等。由虚方法和接口间接结合的对象头,其内存开销通常会很大,哪怕你所需要的只是一个静态类型的数值,也会带来一些编译器的强制操作。有趣的是,在某些情况下,编译器能优化掉一些对象开销,但不总是能起作用。如果你非常在意托管代码的执行效率,那么使用数值或值类型将会有所益处,但在本地C++的类型中,这不算一个很大的区别,当然,C++也没有强制任何编程范式,所以也有可能在C++之上,通过创建库来建立一个这样截然不同的类型系统。
装箱
什么是装箱(boxing)?装箱是一种用来桥接数值和对象的机制。尽管CLR的每种类型都是直接或间接从Object类派生而来,但数值却不是。一个堆栈上的数值(如整形int),只不过是一个编译器会进行某种特定操作的内存块。如果你实在想把一个数值当成一个对象,必须对数值调用从Object继承而来的方法,为了实现这一点,CLR提供了装箱的概念。知道一点装箱的原理还是有点用的,首先,一个数值通过使用ldloc IL指令入栈,接下来,装箱IL指令运行,把数值类型提升,CLR再把数值出栈,并分配足够的空间存储数值与对象头,然后一个对新建对象的引用被压入栈,所有这些就是装箱指令要做的事。最后,为取得对象引用,stloc IL指令从堆栈中弹出引用,并把它存储在局部变量中。
现在,问题是:在编程语言中,对数值的装箱操作,是应该表现为隐式还是显式呢?换句话说,是否应使用一个显式转换或其他构造函数呢?C#语言设计者决定做成隐式转换,毕竟,一个整形数是从Object间接继承来的Int32类型。
问题来了,正如我们所知,装箱不是一个简单的向上转换,它有点像把一个数值转换成一个对象,是一个存在潜在代价的操作。正是因为这个原因,托管C++通过使用关键字__box,来进行显式装箱。
当然,在托管C++中,当装箱一个数值时,不会失去静态类型信息,而这一点,正是C#所缺乏的。
指定强类型的装箱值有利于再次转换回到一个数值类型,或被称为解箱(unboxing),不使用dynamic_cast,只是简单地解引用一个对象。
当然,托管C++的显式装箱所带来的句法上的花销,在许多情况下已被证明是巨大的。正因为此,改变了C++/CLI语言的设计过程,成了与C#保持一致--隐式装箱。在相同情况下,它在直接表示强类型装箱数值上保持了类型安全,而这正是其他 .NET语言所做不到的。
在此,也暗示着一个没有指向任何对象的句柄,不能被初始化为零,在这一点上,与指针是一致的,因为这将导致只是把数值"零"装箱;同时这也是常量nullptr存在的原因,它能被赋给任何句柄,且是C#中关键字null的对等物。尽管在C++/CLI语言设计中,nullptr是一个新的保留字,但它已被Herb Sutter和Bjarne Stroustrup提议增加在标准C++中。
在讨论装箱(boxing)之前,有必要弄清楚为什么值类型与引用类型之间会有所区别。
一个含有数值的值类型的实例,和一个指向对象的引用类型的实例,它们有什么区别呢?除了存储对象所需的内存之外,每一个对象都会有一个对象头,目的是为面向对象的编程提供基本的服务,如存在虚方法的类,嵌入其中的元数据等等。由虚方法和接口间接结合的对象头,其内存开销通常会很大,哪怕你所需要的只是一个静态类型的数值,也会带来一些编译器的强制操作。有趣的是,在某些情况下,编译器能优化掉一些对象开销,但不总是能起作用。如果你非常在意托管代码的执行效率,那么使用数值或值类型将会有所益处,但在本地C++的类型中,这不算一个很大的区别,当然,C++也没有强制任何编程范式,所以也有可能在C++之上,通过创建库来建立一个这样截然不同的类型系统。
装箱
什么是装箱(boxing)?装箱是一种用来桥接数值和对象的机制。尽管CLR的每种类型都是直接或间接从Object类派生而来,但数值却不是。一个堆栈上的数值(如整形int),只不过是一个编译器会进行某种特定操作的内存块。如果你实在想把一个数值当成一个对象,必须对数值调用从Object继承而来的方法,为了实现这一点,CLR提供了装箱的概念。知道一点装箱的原理还是有点用的,首先,一个数值通过使用ldloc IL指令入栈,接下来,装箱IL指令运行,把数值类型提升,CLR再把数值出栈,并分配足够的空间存储数值与对象头,然后一个对新建对象的引用被压入栈,所有这些就是装箱指令要做的事。最后,为取得对象引用,stloc IL指令从堆栈中弹出引用,并把它存储在局部变量中。
现在,问题是:在编程语言中,对数值的装箱操作,是应该表现为隐式还是显式呢?换句话说,是否应使用一个显式转换或其他构造函数呢?C#语言设计者决定做成隐式转换,毕竟,一个整形数是从Object间接继承来的Int32类型。
| int i = 123; object o = i; |
问题来了,正如我们所知,装箱不是一个简单的向上转换,它有点像把一个数值转换成一个对象,是一个存在潜在代价的操作。正是因为这个原因,托管C++通过使用关键字__box,来进行显式装箱。
| int i = 123; Object* o = __box(i); |
当然,在托管C++中,当装箱一个数值时,不会失去静态类型信息,而这一点,正是C#所缺乏的。
| int i = 123; int __gc* o = __box(i); |
指定强类型的装箱值有利于再次转换回到一个数值类型,或被称为解箱(unboxing),不使用dynamic_cast,只是简单地解引用一个对象。
| int c = *o; |
当然,托管C++的显式装箱所带来的句法上的花销,在许多情况下已被证明是巨大的。正因为此,改变了C++/CLI语言的设计过程,成了与C#保持一致--隐式装箱。在相同情况下,它在直接表示强类型装箱数值上保持了类型安全,而这正是其他 .NET语言所做不到的。
| int i = 123; int^ hi = i; int c = *hi; hi = nullptr; |
在此,也暗示着一个没有指向任何对象的句柄,不能被初始化为零,在这一点上,与指针是一致的,因为这将导致只是把数值"零"装箱;同时这也是常量nullptr存在的原因,它能被赋给任何句柄,且是C#中关键字null的对等物。尽管在C++/CLI语言设计中,nullptr是一个新的保留字,但它已被Herb Sutter和Bjarne Stroustrup提议增加在标准C++中。
- 推荐阅讯
- VC++下借助DBGird控件显示数据库记录
- C/C++中命令行参数的原理
- C++启蒙之用C++制作程序
- exit()子程序终止函数与return()函数的差别
- 在VC++中实现自动连续播放多媒体文件
- Windows Sockets:使用 CAsyncSocket 类
- C/C++中利用空指针(NULL),提高程序运行效率
- More Effective C++:不要重载的操作符
- 为C++程序添加文件保存加载功能
- C++运算符重载赋值运算符
- 阅读排行
- 1.Borland 发布C++ Builder 2006 RAD 环境
- 2.C/C++程序员应聘常见面试题深入剖析
- 3.Visual C++常用数据类型转换详解
- 4.C++中的 static 关键字
- 5.利用VC++实现局域网实时视频传输
- 6.浅谈C/C++内存泄漏及其检测工具
- 7.英国投票否决C++/CLI 微软强攻ISO标准受挫
- 8.VC++下用MSComm控件实现串口通讯
- 9.伪随机数生成及在VC++中的实现
- 10.VC++编程实现对波形数据的频谱分析
- 专题教程
- Windows Server-Windows Server文档-Windows Server新闻-Windows Ser PostgreSQL-PostgreSQL文档-PostgreSQL新闻-PostgreSQL专家
- WebLogic-WebLogic文档-WebLogic新闻-WebLogic专家 FreeBSD-FreeBSD文档-FreeBSD新闻-FreeBSD专家
- Linux-内核 GUI KDE Gnome DNS FTP 安全 安装-Linux专区 Windows-AD IIS ServerCore 虚拟化 安全 HPC-Windows专区
- 大话G游 专题:手机病毒揭密
- ARP攻击防范与解决方案 路由故障处理手册
