博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
记一次java中三元表达式的坑(避免踩坑)
阅读量:4035 次
发布时间:2019-05-24

本文共 1394 字,大约阅读时间需要 4 分钟。

近期一直在刷算法,原创文章写的也比较少,今天的主题不算是一个很大的问题,是我做题的时候出来的,而且还曾在A厂的公众号上看到过,今天自己整理一下,避免大家入坑。

这个问题是三元表达式会在计算的时候出现拆箱的运算,造成空指针异常。

一、问题重现

public class Test {
public static void main(String[] args) {
Integer a = null; boolean flag = false; Integer b = flag ? 1*2 : a; System.out.println("程序结束"); }}

代码很简单,a是包装类型Integer,初始值是null,b通过三元表达式进行赋值。运行一下这个代码就会出现空指针异常:

在这里插入图片描述

为什么会出现这个现象呢?下面我们来分析一下:

二、问题分析

刚刚这个空指针现象很容易我们就想到类型转化上,三元表达式的类型转换同样要遵守一定的规则才可以。

1、三元操作符类型的转换规则:

(1)若两个操作数不可转换,则不做转换,返回值为Object类型

比如Object b = flag ? A : B,此时A和B是两个不同的对象,不可转换,那就把最终结果赋值给Object

(2)若两个操作数是明确类型的表达式(比如变量),则按照正常的二进制数字来转换,int类型转换为long类型,long类型转换为float类型等。

这个就比较容易理解了,比如float b = flag ? 1:1.0f。1为int类型向上转即可。

(3)若两个操作数中有一个是数字S,另外一个是表达式,且其类型标示为T,那么,若数字S在T的范围内,则转换为T类型;若S超出了T类型的范围,则T转换为S类型。

*这种情况就是刚刚我们所演示的例子,Integer b = flag ? 1 2: a,结果是b=a,但是a是null,要强制转换成Integer(1 * 2的类型),于是就出现了空指针。

(4)若两个操作数都是直接量数字,则返回值类型为范围较大者。

这种情况和第二种类似。

现在答案基本上出来了,出现空指针的原因是,a=null,要强制转换Integer,于是出现了空指针,因为虚拟机看到一个null就找不到要转化的对象了。

2、反编译分析

现在我们找到Test的字节码文件,输入javap -c命令,反编译一下:

在这里插入图片描述

答案现在应该清楚了,

3、为什么要拆箱(重点)

为什么要对a进行拆箱,直接把a=null赋值给b不就完事了嘛。这一点就需要我们注意一下那个前提条件,也就是说一个是数字,一个是表达式,刚刚的那个例子也验证了这个观点,现在我不是表达式,再来验证一下:

在这里插入图片描述

现在我们可以看到不是表达式,依然会出现这个错误。别着急,还有让你更晕的,我们再来改变一下代码:

在这里插入图片描述

这尼玛恶心,我测试了很多不同的案例,基本上就分为这两种情况。原因如下:

条件表达式?表达式 1 :表达式 2,假设结果是表达式2

(1)表达式1和2都为null,表达式2看到1是null,则不会进行拆箱。

(2)表达式1为数字或者是表达式,表达式2根据1的类型进行拆箱。

也就是说表达式2主要依据表达式1判断是否进行拆箱操作。

三、问题解决

解决方案很简单,那就是只要遇见null在三元表达式里,就尽量转化为if结构。

转载地址:http://vpbdi.baihongyu.com/

你可能感兴趣的文章
Idea导入的工程看不到src等代码
查看>>
技术栈
查看>>
Jenkins中shell-script执行报错sh: line 2: npm: command not found
查看>>
8.X版本的node打包时,gulp命令报错 require.extensions.hasownproperty
查看>>
Jenkins 启动命令
查看>>
Maven项目版本继承 – 我必须指定父版本?
查看>>
Maven跳过单元测试的两种方式
查看>>
通过C++反射实现C++与任意脚本(lua、js等)的交互(二)
查看>>
利用清华镜像站解决pip超时问题
查看>>
[leetcode BY python]1两数之和
查看>>
微信小程序开发全线记录
查看>>
Centos import torchvision 出现 No module named ‘_lzma‘
查看>>
PTA:一元多项式的加乘运算
查看>>
CCF 分蛋糕
查看>>
解决python2.7中UnicodeEncodeError
查看>>
小谈python 输出
查看>>
Django objects.all()、objects.get()与objects.filter()之间的区别介绍
查看>>
python:如何将excel文件转化成CSV格式
查看>>
Django 的Error: [Errno 10013]错误
查看>>
机器学习实战之决策树(一)
查看>>