时间:2021-05-20
最近修改线上bug的时候排查了一个十分隐藏的bug,直接上代码:
Integer a = null;boolean flag = true;Integer b = flag ? a : 0;乍一看是没什么毛病的,但是已运行就会发现报空指针,在idea里面也会警告可能有空指针,这是什么原因呢?
直接看字节码:
0: aconst_null1: astore_12: iconst_13: istore_24: iload_25: ifeq 158: aload_19: invokevirtual #2 // Method java/lang/Integer.intValue:()I12: goto 1615: iconst_016: invokestatic #3 // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;19: astore_320: getstatic #4 // Field java/lang/System.out:Ljava/io/PrintStream;23: aload_324: invokevirtual #5 // Method java/io/PrintStream.println:(Ljava/lang/Object;)V27: return可以看到字节码中调用了`Integer.valueOf()`方法,因为我们代码中一个值使用的是0(基本数据类型int),编译器就会进行自动拆装箱(成int),
虽然三目运算的后面逻辑不会执行,但是隐藏的自动拆装箱会执行`Integer.valueOf()`方法,也就有了空指针异常。
为了进一步验证存在自动拆装箱,把代码修改一下:
Integer a = null;boolean flag = true;Integer b = flag ? a : new Integer(0);再看字节码:
0: aconst_null1: astore_12: iconst_13: istore_24: iload_25: ifeq 128: aload_19: goto 2012: new #2 // class java/lang/Integer15: dup16: iconst_017: invokespecial #3 // Method java/lang/Integer."<init>":(I)V20: astore_321: getstatic #4 // Field java/lang/System.out:Ljava/io/PrintStream;24: aload_325: invokevirtual #5 // Method java/io/PrintStream.println:(Ljava/lang/Object;)V可以看到,由于重新创建了一个`Integer`对象,并没有基本类型的存在,也就不存在自动拆装箱,修改过后的代码也就不会有问题了,但是idea的警告依旧存在。
这是一个非常隐蔽,也非常容易忽略和踩坑的一个地方,三目运算符的使用应该保证后面的值都是常量,或者统一类型,不然就会出现上面的情况。
更甚三目运算符本身提供的作用也不过是为了简化逻辑,在其中放入过多的逻辑判断也就违背了其初衷。
总结
以上所述是小编给大家介绍的Java三目运算中隐藏的自动拆装箱,希望对大家有所帮助,如果大家有任何疑问欢迎给我留言,小编会及时回复大家的!
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
下面说的和三目运算有点相似,但又不一样,实在不知道该如何拟定标题,先就是这个标题吧,大家都知道python中没有三目运算,但是and/or有点类似三目运算:an
在面试过程中,常常会有面试官问到基础的问题的时候都会问到Java的拆装箱,关于这个问题其实不是很难,但是如果平时进行自学的时候不是注意,就可能一脸懵逼,所以笔者
三目运算符也就是三元运算符一些语言(如Java)的三元表达式形如:判定条件?为真时的结果:为假时的结果result=xifxPython的三元表达式有如
什么是自动装箱和拆箱自动装箱就是Java自动将原始类型值转换成对应的对象,比如将int的变量转换成Integer对象,这个过程叫做装箱,反之将Integer对象
问题:如下代码: 原因:由于有两个三目运算,所以第二个三目运算要加上括号。 正确写法: