Java三目运算中隐藏的自动拆装箱

网友投稿 209 2023-07-17

Java三目运算中隐藏的自动拆装箱

最近修改线上bug的时候排查了一个十分隐藏的bug,直接上代码:

Integer a = null;

boolean flag = true;

Integer b = flag ? a : 0;

乍一看是没什么毛病的,但是已运行就会发现报空指针,在idea里面也会警告可能有空指针,这是什么原因呢?

直接看字节码:

0: aconst_null

1: astore_1

2: iconst_1

3: istore_2

4: iload_2

5: ifeq 15

8: aload_1

9: invokevirtual #2 // Method java/lang/Integer.intValue:()I

12: goto 16

15: iconst_0

16: invokestatic #3 // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;

19: astore_3

20: getstatic #4 // Field java/lang/System.out:Ljava/io/PrintStream;

23: aload_3

24: invokevirtual #5 // Method java/io/PrintStream.println:(Ljava/lang/Object;)V

27: return

可以看到字节码中调用了`Integer.valueOf()`方法,因为我们代码中一个值使用的是0(基本数据类型int),编译器就会进行自动拆装箱(成int),

虽然三目运算的后面逻辑不会执行,但是隐藏的自动拆装箱会执行`Integer.valueOf()`方法,也就有了空指针异常。

为了进一步验证存在自动拆RPQCUZz装箱,把代码修改一下:

Integer a = null;

boolean flag = true;

Integer b = flag ? a : new Integer(0);

再看字节码:

0: aconst_null

1: astore_1

2: iconst_1

3: istore_2

4: iload_2

5: ifeq 12

8: aload_1

9: goto 20

12: new #2 // class java/lang/Integer

15: dup

16: iconst_0

17: invokespecial #3 // Method java/lang/Integer."":(I)V

20: astore_3

21: getstatic #4 // Field java/lang/System.out:Ljava/io/PrintStream;

24: aload_3

25: invokevirtual #5 // Method java/io/PrintStream.println:(Ljava/lang/Object;)V

可以看到,由于重新创建了一个`Integer`对象,并没有基本类型的存在,也就不存在自动拆装箱,修改过后的代码也就不会有问题了,但是idea的警告依旧存在。

这是一个非常隐蔽,也非常容易忽略和踩坑的一个地方,三目运算符的使用应该保证后面的值都是常量,或者统一类型,不然就会出现上面的情况。

更甚三目运算符本身提供的作用也不过是为了简化逻辑,在其中放入过多的逻辑判断也就违背了其初衷。

总结

以上所述是给大家介绍的Java三目运算中隐藏的自动拆装箱,希望对大家有所帮助,如果大家有任何疑问欢迎给我留言,会及时回复大家的!

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:Java中初始化List的5种方法示例
下一篇:Mybatis核心组成部分之SQL映射文件揭秘详解
相关文章

 发表评论

暂时没有评论,来抢沙发吧~