一分钟理解Java包装类型

Java 一直标榜自己是一个纯粹的面向对象语言, 自作聪明的为所有的值类型都提供相应的引用类型(不明白这两个概念,看之前的《一分钟理解传值和传引用》) 比如:int 类型对应的有 Integer,前者是一个值,后者是一个引用。 为了方便二者的转换又一个叫“自动拆装箱”的特性,把本来清晰的概念搞的乱七八糟。

一个优秀的语言应该语法简单,语义单一、清晰。

本文讨论它这些乌七八糟的概念(我也搞不懂),直接进入正题——通过阅读 JVM code 判断究竟发生了什么。

解读 class 文件

JVM 是一个栈式虚拟机,它提供的指令都是围绕着栈进行的。通过 javap -c 查看一个 class 文件中的 JVM 指令。 如下代码,左边是 Java 代码右边是它的 JVM code。

13465705-61011da6bc8518df.jpg

看一下每条指令执行完后栈的变化: bipush 把数字 20 直接 push 到栈

13465705-d7ff20ef0a8624f1.png

invokestatic 调用一个静态方法在中构造一个对象,然后把对象的地址压入到栈

13465705-b1b4029738b18f6a.png

astore_1 把 Integer 对象的内存地址记录到一个内部变量中(JVM 在堆中维护了一张大的变量表,代表变量名和变量值的关系,可以想象成 HashMap。)至此,Integer = 20 这句代码执行完毕。紧接着看,bipush 把 10 压入栈asotre2 把变量 b 和栈中的 10 做关联(放到变量表中)

13465705-bef3e88142028da6.jpg
13465705-9dfc631431a07edc.png
13465705-cb48aa62059b14b7.png

值变量所指向的内容(值)是放在栈中的,访问时直接操作栈

引用变量所指向的内容(对象)是放在堆中的,访问时先把变量载入到栈(通过aload_1 指令,例子中没有出现),再操作。

访问包装对象时发生了什么

Integerc = null;Integerd =10;inte = c + d;

JVM code 为

13465705-694bdd2ce3f881fb.jpg

包装对象的空指针问题

aconst_null 把一个空指针压入栈,astore_1 把栈顶的变量放入到变量表中,所以此时 a 是 null,所以会出现空指针错误。

包装对象的计算方法

8-16 是计算两数相加,aload_1 把变量表中的变量压入栈,invokervirtual 指令把对象转换成 int 重新入栈;12、13 行的逻辑也是如此。 16 行执行整数相加。 因为计算结果是 int 类型,所以最后通过 isotre_3 放到变量表。

自己分析

如果代码的最后一行写作 Integer e = c + d; ,JVM code 会变成

13465705-4f4412ee3c8f5da5.jpg

自己动手分析一下看看吧。

Java 的包装数据类型非常蹩脚,这是它为了追求“表面的面向对象”而付出的代价。装逼之势如雷霆万钧,可怕。

欢迎工作一到五年的Java工程师朋友们加入Java进阶高级架构:855355016

本群提供免费的学习指导 架构资料 以及免费的解答

不懂得问题都可以在本群提出来 之后还会有职业生涯规划以及面试指导

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值