当前位置: 首页 > news >正文

JAVA自动装箱拆箱


引言

Java 中的**装箱(Boxing)拆箱(Unboxing)**是自动类型转换的机制,用于在基本数据类型(如 intlong 等)和其对应的包装类(如 IntegerLong 等)之间进行转换。这种机制简化了代码的编写,但也可能引发一些性能问题或意外行为。

通过以下案例和字节码分析,我们将深入探讨装箱和拆箱的原理及其在实际开发中的应用。


案例代码与输出

public class Test {public static void main(String[] args) {Integer a = 1;          // 自动装箱Integer b = 2;          // 自动装箱Integer c = 3;          // 自动装箱Integer d = 3;          // 自动装箱Integer e = 321;        // 自动装箱Integer f = 321;        // 自动装箱Long g = 3L;            // 自动装箱System.out.println(c == d);           // trueSystem.out.println(e == f);           // falseSystem.out.println(c == (a + b));     // trueSystem.out.println(c.equals(a + b));  // trueSystem.out.println(g == (a + b));     // trueSystem.out.println(g.equals(a + b));  // false}
}
输出结果:
true
false
true
true
true
false

逐行解析与字节码分析

1. c == d 输出 true
  • 原因cd 都是 Integer 类型,值为 3
  • 装箱过程Integer c = 3 实际上被编译器翻译为 Integer c = Integer.valueOf(3)
  • 缓存机制Integer.valueOf 方法会对 -128 ~ 127 范围内的整数使用缓存池。因此,cd 指向同一个缓存对象。
  • 字节码分析
     ICONST_3INVOKESTATIC java/lang/Integer.valueOf (I)Ljava/lang/Integer;ASTORE 3ICONST_3INVOKESTATIC java/lang/Integer.valueOf (I)Ljava/lang/Integer;ASTORE 4
    
    • ASTORE 3ASTORE 4 分别存储 cd
    • 因为 3 在缓存范围内,cd 引用的是同一个对象,因此 c == d 返回 true

2. e == f 输出 false
  • 原因ef 的值为 321,超出了 Integer 缓存范围(默认 -128 ~ 127),因此每次调用 Integer.valueOf(321) 都会创建新的对象。
  • 字节码分析
     SIPUSH 321INVOKESTATIC java/lang/Integer.valueOf (I)Ljava/lang/Integer;ASTORE 5SIPUSH 321INVOKESTATIC java/lang/Integer.valueOf (I)Ljava/lang/Integer;ASTORE 6
    
    • SIPUSH 将常量 321 压入栈顶。
    • INVOKESTATIC 调用 Integer.valueOf 方法。
    • 因为 321 不在缓存范围内,ef 是不同的对象,因此 e == f 返回 false

3. c == (a + b) 输出 true
  • 原因a + b 的计算涉及拆箱操作,a.intValue()b.intValue() 相加得到一个 int 值,然后与 c 进行比较时,c 也会被拆箱为 int
  • 拆箱过程
    • a + b 被翻译为 a.intValue() + b.intValue()
    • c == (a + b) 被翻译为 c.intValue() == (a.intValue() + b.intValue())
  • 字节码分析
     ALOAD 3INVOKEVIRTUAL java/lang/Integer.intValue ()IALOAD 1INVOKEVIRTUAL java/lang/Integer.intValue ()IALOAD 2INVOKEVIRTUAL java/lang/Integer.intValue ()IIADDIF_ICMPNE L14
    
    • INVOKEVIRTUAL 调用 intValue 方法完成拆箱。
    • 最终比较的是两个 int 值,因此 c == (a + b) 返回 true

4. c.equals(a + b) 输出 true
  • 原因equals 方法比较的是值,而不是引用。a + b 的结果是一个 int,会被自动装箱为 Integer,然后调用 equals 方法进行比较。
  • 字节码分析
     ALOAD 3ALOAD 1INVOKEVIRTUAL java/lang/Integer.intValue ()IALOAD 2INVOKEVIRTUAL java/lang/Integer.intValue ()IIADDINVOKESTATIC java/lang/Integer.valueOf (I)Ljava/lang/Integer;INVOKEVIRTUAL java/lang/Integer.equals (Ljava/lang/Object;)Z
    
    • a + b 的结果被装箱为 Integer
    • equals 方法比较的是两个 Integer 的值,因此返回 true

5. g == (a + b) 输出 true
  • 原因gLong 类型,a + b 的结果是 int 类型。在比较时,a + b 被提升为 long 类型(I2L 指令),然后与 g 的值进行比较。
  • 字节码分析
     ALOAD 7INVOKEVIRTUAL java/lang/Long.longValue ()JALOAD 1INVOKEVIRTUAL java/lang/Integer.intValue ()IALOAD 2INVOKEVIRTUAL java/lang/Integer.intValue ()IIADDI2LLCMPIFNE L18
    
    • I2Lint 提升为 long
    • LCMP 比较两个 long 值,因此 g == (a + b) 返回 true

6. g.equals(a + b) 输出 false
  • 原因equals 方法比较的是对象类型和值。a + b 的结果是 int 类型,会被装箱为 Integer,而 gLong 类型,因此 equals 返回 false
  • 字节码分析
     ALOAD 7ALOAD 1INVOKEVIRTUAL java/lang/Integer.intValue ()IALOAD 2INVOKEVIRTUAL java/lang/Integer.intValue ()IIADDINVOKESTATIC java/lang/Integer.valueOf (I)Ljava/lang/Integer;INVOKEVIRTUAL java/lang/Long.equals (Ljava/lang/Object;)Z
    
    • a + b 的结果被装箱为 Integer
    • equals 方法检查类型不匹配(Long vs Integer),因此返回 false

总结与注意事项

  1. 装箱与拆箱的本质

    • 装箱:将基本类型(如 int)转换为包装类(如 Integer)。底层调用 valueOf 方法。
    • 拆箱:将包装类(如 Integer)转换为基本类型(如 int)。底层调用 intValue 方法。
  2. 缓存机制的影响

    • Integer 的缓存范围是 -128 ~ 127,超出范围会创建新对象。
    • 可以通过 JVM 参数(如 -XX:AutoBoxCacheMax=512)调整缓存范围。
  3. 比较操作的陷阱

    • 使用 == 比较引用类型时,可能会因为缓存或对象创建方式不同而导致结果不符合预期。
    • 推荐使用 equals 方法进行值比较。
  4. 性能问题

    • 频繁的装箱和拆箱操作会导致额外的对象创建和方法调用,影响性能。
    • 在性能敏感的场景下,尽量避免不必要的装箱和拆箱。

扩展思考

在前面的分析中,我们提到 e == f 的结果为 false,因为 321 超出了 Integer 默认的缓存范围(-128 ~ 127),导致每次调用 Integer.valueOf(321) 都会创建新的对象。然而,Java 提供了一种方式来扩展 Integer 的缓存范围,从而改变这一行为。

如何调整缓存范围
Integer 的缓存范围可以通过 JVM 参数 -XX:AutoBoxCacheMax= 进行调整。例如,如果我们希望将缓存范围扩展到 512,可以在启动 JVM 时添加以下参数:

java -XX:AutoBoxCacheMax=512 Test

调整后的效果
当我们将缓存范围扩展到 512 后,e == f 的结果会发生变化:

原因:321 现在位于缓存范围内,因此 Integer.valueOf(321) 会返回缓存中的同一个对象。
输出结果:true

底层原理分析
通过调整缓存范围,Integer.valueOf 方法的行为发生了变化:

如果值在缓存范围内(-128 ~ AutoBoxCacheMax),则返回缓存中的对象。
如果值超出缓存范围,则创建新的 Integer 对象。
以下是 Integer.valueOf 方法的源码片段:

public static Integer valueOf(int i) {if (i >= IntegerCache.low && i <= IntegerCache.high)return IntegerCache.cache[i + (-IntegerCache.low)];return new Integer(i);
}

其中,IntegerCache.low 和 IntegerCache.high 分别表示缓存范围的下限和上限。默认情况下,IntegerCache.low = -128,IntegerCache.high = 127。通过 JVM 参数 -XX:AutoBoxCacheMax,我们可以动态调整 IntegerCache.high 的值。

相关文章:

  • make和makefile的使用,以及写一个简单的进度条程序
  • DAMA第10章深度解析:参考数据与主数据管理的核心要义与实践指南
  • 挪度半身复苏小安妮模型QCPR成人半身急救心肺复苏模拟人
  • 使用python脚本连接SQL Server数据库导出表结构
  • “AI+城市治理”智能化解决方案
  • Profinet转CanOpen协议转换网关,破解工业设备“语言障碍”
  • 计算机图形学编程(使用OpenGL和C++)(第2版)学习笔记 08.阴影
  • Vue.js框架的优缺点
  • 【免费工具】图吧工具箱2025.02正式版
  • Discriminative and domain invariant subspace alignment for visual tasks
  • GD32H7复位后程序调用函数时间增加
  • 第八周作业
  • 基于STM32的LCD信号波形和FFT频谱显示
  • “睿思 BI” 系统介绍
  • 自学嵌入式 day 17- c语言-第11章 结构体与共用体 第12章 位运算
  • 批量导出docker镜像
  • 如何解决 PowerShell 显示 “此系统上禁用了脚本运行” 的问题
  • 在资源受限设备上实现手势识别:基于包络EMG数据和实时测试的Tiny-ML方法
  • 【Linux】用户管理
  • Dify-1.3.1介绍及部署镜像下载
  • 学习时报头版:世界要公道不要霸道
  • 7月打卡乐高乐园,还可以去千年古镇枫泾参加这个漫画艺术季
  • 重庆三峡学院回应“中标价85万设备网购300元”:已终止采购
  • 西甲上海足球学院揭幕,用“足球方法论”试水中国青训
  • 聆听百年唐调正声:唐文治王蘧常吟诵传习的背后
  • 习近平向中国人民解放军仪仗队致意