内容简介:最近修改线上bug的时候排查了一个十分隐藏的bug,直接上代码:乍一看是没什么毛病的,但是已运行就会发现报空指针,在idea里面也会警告可能有空指针,这是什么原因呢?直接看字节码:
最近修改线上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()`方法,也就有了空指针异常。
为了进一步验证存在自动拆装箱,把代码修改一下:
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."<init>":(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的警告依旧存在。
这是一个非常隐蔽,也非常容易忽略和踩坑的一个地方,三目运算符的使用应该保证后面的值都是常量,或者统一类型,不然就会出现上面的情况。
更甚三目运算符本身提供的作用也不过是为了简化逻辑,在其中放入过多的逻辑判断也就违背了其初衷。
BugHome版权所有丨转载请注明出处:https://minei.me/archives/422.html
以上所述就是小编给大家介绍的《Java三目运算中隐藏的自动拆装箱》,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对 码农网 的支持!
本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。
文明之光(第一册)
吴军 / 人民邮电出版社 / 2014-6-25 / 59.00元
人类的历史,是从野蛮蒙昧一步步走向文明进步的过程。在文明的进程中,人类创造出多元的文化,它们有着各自的特长。要实现人类和平发展的终极理想,一个重要的前提是承认文化的多元性,并且取长补短,相互融合。 吴军博士写作《文明之光》系列,希望能开阔人们的视野,让我们看到各种各样的人类文明。虽然今天不同的地区发达程度不同,文明历史的长短不一,国家亦有大小之分,但是文明之光从世界的每一个角落发出,对人类的......一起来看看 《文明之光(第一册)》 这本书的介绍吧!