[019] C#基础:理解装箱与拆箱

.NET大牛之路 • 王亮@精致码农 • 2021.08.27

前面我们讲到 .NET 平台支持的两大数据类型:值类型和引用类型。值类型比引用类型更高效,因为它没有指针引用,不用分配在托管堆中,也不用被 GC 回收。但有时候你可能偶尔需要将一种类型的变量表示为另一种类型的变量。为此,C# 提供了装箱拆箱的机制。

1理解装箱

简单地说,装箱就是将一个值类型的数据存储在一个引用类型的变量中。

假设你一个方法中创建了一个 int 类型的本地变量,你要将这个值类型表示为一个引用类型,那么就表示你对这个值进行了装箱操作,如下所示:

static void SimpleBox()
{int myInt = 25;// 装箱操作object boxedInt = myInt;
}

确切地说,装箱的过程就是将一个值类型分配给 Object 类型变量的过程。当你装箱一个值时,CoreCLR 会在堆上分配一个新的对象,并将该值类型的值复制到该对象实例。返回给你的是一个在托管堆中新分配的对象的引用。

2理解拆箱

反过来,将 Object 引用类型变量的值转换回栈中相应的值类型的过程则称为拆箱

从语法上讲,拆箱操作看起来就像一个正常的转换操作。然而,其语义是完全不同的。CoreCLR 首先验证接收的数据类型是否等同于被装箱的类型,如果是,它就把值复制回基于栈存储的本地变量中。

例如,如果下面的 boxedInt 的底层类型确实是 int,那就完成了拆箱操作:

static void SimpleBoxUnbox()
{int myInt = 25;// 装箱操作object boxedInt = myInt;// 拆箱操作int unboxedInt = (int)boxedInt;
}

记住,与执行典型的类型转换不同,你必须将其拆箱到一个恰当的数据类型中。如果你试图将一块数据拆箱到不正确的数据类型中,将会抛出 InvalidCastException 异常。为了安全起见,如果你不能保证 Object 类型背后的类型,最好使用 try/catch 逻辑把拆箱操作包起来,尽管这样会有些麻烦。考虑下面的代码,它将抛出一个错误,因为你正试图将装箱的 int 类型拆箱成一个 long 类型:

static void SimpleBoxUnbox()
{int myInt = 25;// 装箱操作object boxedInt = myInt;// 拆箱到错误的数据类型,将触发运行时异常try{long unboxedLong = (long)boxedInt;}catch (InvalidCastException ex){Console.WriteLine(ex.Message);}
}

3生成的 IL 代码

当 C# 编译器遇到装箱/拆箱语法时,它会生成包含装箱/拆箱操作的 IL 代码。如果你用 ildasm.exe 查看编译的程序集,你会看到装箱和拆箱操作对应的 boxunbox 指令:

.method assembly hidebysig staticvoid  '<<Main>$>g__SimpleBoxUnbox|0_0'() cil managed
{.maxstack  1.locals init (int32 V_0, object V_1, int32 V_2)IL_0000:  nopIL_0001:  ldc.i4.s   25IL_0003:  stloc.0IL_0004:  ldloc.0IL_0005:  box        [System.Runtime]System.Int32IL_000a:  stloc.1IL_000b:  ldloc.1IL_000c:  unbox.any  [System.Runtime]System.Int32IL_0011:  stloc.2IL_0012:  ret} // end of method '<Program>$'::'<<Main>$>g__SimpleBoxUnbox|0_0'

乍一看,装箱/拆箱似乎是一个没啥用的语言特性,学术性大于实用性。毕竟,你很少需要在一个本地 Object 变量中存储一个本地值类型。然而,事实是装箱/解箱过程是相当有用的,因为它允许你假设一切都可以被当作 Object 类型来处理,而 CoreCLR 会自动帮你处理与内存有关的细节。

4实际应用

让我们来看看装箱/拆箱的实际应用,我们以 C# 的 ArrayList 类为例,用它来保存一批在栈中存储的整型数据。ArrayList 类的相关方法成员列举如下:

public class ArrayList : IList, ICloneable
{...public virtual int Add(object? value);public virtual void Insert(int index, object? value);public virtual void Remove(object? obj);public virtual object? this[int index] { get; set; }
}

请注意,上面 ArrayList 的方法都是对 Object 类型数据进行操作。ArrayList 是为操作对象(代表任何类型)而设计的,而对象是在托管堆上分配的数据。请考虑下面代码:

static void WorkWithArrayList()
{// 当传递给对象的方法时,值类型会自动被装箱ArrayList myInts = new ArrayList();myInts.Add(10);
}

尽管你直接将数字数据传入需要 Object 参数的方法中,但运行时自动将分配在栈中的数据装箱。如果你想使用索引器从 ArrayList 中检索一条数据,你必须使用转换操作将堆分配的对象拆箱为栈分配的整型,因为 ArrayList 的索引器返回的是 Object 类型,而不是 int 类型。

static void WorkWithArrayList()
{// 当传递给需要对象参数的方法时,值类型就自动被装箱ArrayList myInts = new ArrayList();myInts.Add(10);// 当对象被转换回基于栈存储的数据时,就会发生拆箱int i = (int)myInts[0];// 由于 WriteLine() 需要的 object 参数,又重新装箱了Console.WriteLine("Value of your int: {0}", i);
}

在调用 ArrayList.Add() 之前,在栈中分配的 int 数值被装箱了,所以它可以被传入参数为 Object 类型的方法中。从 ArrayList 中检索到 Object 类型的数据时,通过转换操作,它就被拆箱成 int 类型。最后,当它被传递给 Console.WriteLine() 方法时,又被装箱了,因为这个方法的参数是 Object 类型。

5小结

从程序员的角度来看,装箱和拆箱是很方便的,我们不需要手动去复制和转移内存中的值类型和引用类型的数据。

但装箱和拆箱背后的栈/堆内存转移也带来了性能问题。下面总结一下对一个简单的整型数进行装箱和拆箱所需要的步骤:

  1. 在托管堆中分配一个新对象;

  2. 在栈中的数据值被转移到该托管堆中的对象上;

  3. 当拆箱时,存储在堆中对象上的值被转移回栈中;

  4. 堆上未使用的对象将最终被 GC 回收。

尽管很多时候装箱和拆箱操作不会在性能方面造成重大影响,但如果一个像 ArrayList 这样的集合包含成千上万条数据,而你的程序又会频繁操作这些数据,性能的影响还是会很明显的。

所以,我们平时在编程时应当尽量避免发生装箱和拆箱操作。比如对于上面 ArrayList 的示例,如果集合元素类型是一致的,则应当使用泛型的集合类型,比如改用 List<T>LinkedList<T> 等。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/298385.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

H3C 5510 交换机DHCP设置

DHCP不能发现网络上非DHCP客户机已经在使用的IP地址&#xff1b;当网络上存在多个DHCP服务器时&#xff0c;一个DHCP服务器不能查出已被其它服务器租出去的IP地址&#xff1b;DHCP服务器不能跨路由器与客户机通信&#xff0c;除非路由器允许BOOTP转发。PC发出的广播包&#xff…

opentrace在mysql中使用_采用OpenReplicator解析MySQL binlog

Open Replicator是一个用Java编写的MySQL binlog分析程序。Open Replicator 首先连接到MySQL(就像一个普通的MySQL Slave一样)&#xff0c;然后接收和分析binlog&#xff0c;最终将分析得出的binlog events以回调的方式通知应用。Open Replicator可以被应用到MySQL数据变化的实…

雷军:有人说我写的代码像诗一样优雅~

全世界只有3.14 % 的人关注了爆炸吧知识整合整理&#xff1a;程序员的那些事&#xff08;id&#xff1a;iProgrammer&#xff09;雷军的代码像诗一样优雅↓↓↓有些网友在评论中质疑&#xff0c;说雷军代码不会是「屎」一样优雅吧。说这话的网友&#xff0c;也许是开玩笑的&…

国外网站评出对程序员最具影响的书籍清单

国外知名网站 stackoverflow 上有一个问题调查&#xff1a; 哪本书是对程序员最有影响、每个程序员都该阅读的书&#xff1f;这个调查已历时两年&#xff0c;目前为止吸引了153,432 人访问&#xff0c;读者共推荐出了 478 本书(还在增加)&#xff0c;其中最火的一本书《Code Co…

python大于小于_在Python中大于/小于Pandas DataFrames / Series之间的比较

如何在DataFrame和Series之间进行比较&#xff1f;我想掩盖DataFrame / Series中比其他DataFrame / Series中的元素更大/更小的元素. 例如,以下内容不会替换大于均值的元素 与nans虽然我期待它&#xff1a; >>> x pd.DataFrame(data{a: [1, 2], b: [3, 4]}) >>…

NodeJS学习笔记

通过js创建个简单的web服务器 var httprequire(http); http.createServer(function(req,res){ res.writeHead(200,{Content-Type:text/html}); res.end("server is up!"); }).listen(8000); console.log(listened on 8000); 推荐学习:Node入门 转载于:https://www.cn…

mysql分析日志_MYSQL 索引(三)--- SQL日志分析

慢查询日志Mysql 的慢查询日志是 Myql 提供的一种日志记录&#xff0c;用来记录在 Myql 中响应时间查过阈值的语句&#xff0c;具体指运行时间超过 long_query_time 值的 SQL&#xff0c;则会被记录在日志中。long_query_time 默认为 10&#xff0c;单位为秒。默认情况下&#…

【转】SMIL基础教程(1)

最近公司项目需要使用到smil相关知识&#xff0c;因而专门学习了一下。在网上找到了几篇基础教程&#xff0c;转载以方便查看。一、 简介随着流技术的成熟和广泛的应用&#xff0c;其优点我们有了深深的体会。但是&#xff0c;其不足之处也逐渐体现出来。问题的出现&#xff0c…

C#多线程开发-任务并行库

你好&#xff0c;我是阿辉。正文共2090字&#xff0c;预计阅读时间&#xff1a;6min。之前学习了线程池&#xff0c;知道了它有很多好处。使用线程池可以使我们在减少并行度花销时节省操作系统资源。可认为线程池是一个抽象层&#xff0c;其向程序员隐藏了使用线程的细节&#…

.cue 文件格式

cue文件格式&#xff08;基础版&#xff09; PERFORMER "陈小春" TITLE "抱一抱" FILE "陈小春.-.[抱一抱].专辑.(ape).ape" WAVE TRACK 01 AUDIO TITLE "抱一抱" INDEX 01 00:00:00 TRACK 02 AUDIO TITLE "我爱的人…

python切片原理_深度解析Python切片

详解Python 切片语法 Python的切片是特别常用的功能&#xff0c;主要用于对列表的元素取值。使用切片也会让你的代码显得特别Pythonic。 切片的主要声明如下&#xff0c;假设现在有一个list&#xff0c;命名为alist&#xff1a; alist [0,1,2,3,4] 切片语法的基本形式为&#…

为什么数学不好,和语文有关系?

▲ 点击查看苏步青教授在担任复旦大学校长时曾经说过:“如果允许复旦大学单独招生考试&#xff0c;我的意见是第一堂课就考语文&#xff0c;考后就批卷子。不合格的&#xff0c;以下的功课就不要考了。语文你都不行&#xff0c;别的是学不通的。”苏步青作为享誉世界的数学家&a…

Docker 博客

Docker 常用命令&#xff1a;首先推荐&#xff1a;http://blog.tankywoo.com/docker/2014/05/08/docker-4-summary.html Docker 网络桥接&#xff1a;http://blog.tankywoo.com/2014/12/22/docker-bridge-network.html docker 大牛的博客&#xff1a;http://blog.csdn.net/smal…

python string length_如何使用python获取字符串长度?哪些方法?

掌握多种python技巧&#xff0c;对于我们更好的灵活应用python是非常重要的&#xff0c;比如接下来给大家介绍的获取字节长度&#xff0c;那大家脑海里就该有印象了&#xff0c;有几种方法呢&#xff1f;一起来看下吧~1、使用len()函数这是最直接的方法。 在这里&#xff0c;我…

二进制、八进制、十进制、十六进制之间转换

一、 十进制与二进制之间的转换 &#xff08;1&#xff09; 十进制转换为二进制&#xff0c;分为整数部分和小数部分 ① 整数部分 方法&#xff1a;除2取余法&#xff0c;即每次将整数部分除以2&#xff0c;余数为该位权上的数&#xff0c;而商继续除以2&#xff0c;余数又为上…

【招聘(北京成都)】北森 招聘.NET 架构师工程师

.net后端架构师 25k-38k14薪工作职责:1.根据业务框架要求&#xff0c;提供技术实现方案&#xff1b;2.负责技术架构选型、并主导功能模块设计、数据结构设计、对外接口设计&#xff1b;3.负责核心技术攻关、系统调优&#xff0c;使系统架构、代码结构不断演进优化&#xff1b;4…

android 网络开发

反复研究了 Android Market&#xff0c;总结一下&#xff0c;之前发在新浪微博上&#xff0c;但不够详细&#xff0c;主要是提高用户体验。 1.网络异常处理&#xff0c;重试机制。 上wifi常常网络断开&#xff0c;那就看运用程序是否健壮。可以用Android 提供的 frameworks/bas…

python turtle 绘图_谈一下Pycharm中关联系统Python解释器的方法

大家知道&#xff0c;PyCharm是一款著名的Python IDE开发工具&#xff0c;是拥有一整套可以帮助用户在使用Python语言开发时提高其效率的工具&#xff0c;具备基本的调试、语法高亮、Project管理、代码跳转、智能提示、自动完成、单元测试、版本控制等等。该IDE分为社区免费版(…

python中线程和进程_python中线程和进程的简单了解

一、操作系统、应用程序1.硬件&#xff1a;硬盘、cpu、主板、显卡........2.装系统(本身也是一个软件)&#xff1a;系统就是一个由程序员写出来的软件&#xff0c;该软件用于控制计算机得硬盘&#xff0c;让他们之间进行互相配合。3.安装软件&#xff1a;各种应用程序二、并发和…

一份干货满满的PPT,答辩加分手到擒来!

全世界只有3.14 % 的人关注了爆炸吧知识对很多童鞋来说&#xff0c;PPT可以说是使用频率很高的办公软件了。毕业答辩需要PPT&#xff0c;项目总结需要PPT&#xff0c;演讲也都需要PPT……可你是否因为PPT陷入这样的——脑壳疼状态&#xff01;花费大量时间&#xff0c;结果PPT效…