IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> Java知识库 -> java == 引发的一个线上异常 -> 正文阅读

[Java知识库]java == 引发的一个线上异常

今天分享遇到的一个线上的 bug,线上代码:

class Scratch {
    public static void main(String[] args) {
        JSONArray arrays = JSONUtil.parseArray("[{'type':1},{},{'type':2},{'type':2}" +
                ",{'name':'zhangsan'},{'type':1},{'type':1},{'type':1}]");
        List<User> users = JSONUtil.toList(arrays, User.class);
        Set<User> set = users.stream().filter(u -> u.getType() == 1).collect(Collectors.toSet());
        System.out.println(set);
    }
    @Data
    static class User {
        private String name;
        private Integer type;
    }
}

类似于这样子的一段代码会抛出一个空指针异常,你可以尝试找一下哪里有可能会出现空指针异常。
异常堆栈长这样子:

Exception in thread "main" java.lang.NullPointerException
	at Scratch.lambda$main$0(scratch.java:14)
	at java.util.stream.ReferencePipeline$2$1.accept(ReferencePipeline.java:174)
	at java.util.ArrayList$ArrayListSpliterator.forEachRemaining(ArrayList.java:1382)
	at java.util.stream.AbstractPipeline.copyInto(AbstractPipeline.java:481)
	at java.util.stream.AbstractPipeline.wrapAndCopyInto(AbstractPipeline.java:471)
	at java.util.stream.ReduceOps$ReduceOp.evaluateSequential(ReduceOps.java:708)
	at java.util.stream.AbstractPipeline.evaluate(AbstractPipeline.java:234)
	at java.util.stream.ReferencePipeline.collect(ReferencePipeline.java:499)
	at Scratch.main(scratch.java:14)

这个空指针异常还是比较好找到的,位于 Stream 中的 filter 中比较出现了异常:

u -> u.getType() == 1

我一开始的想法是对象 u 是一个 null 但后来发现不是,最终找到的地方是 u.getType() 是一个null,是由于 null == 1 抛出了一个空指针异常。

这就涉及到一个 java 的基础点了 null == 1 等于什么?

==java 中一个双目比较运算符,可以用于基础数据类型和引用数据类型的比较,当基础数据类型之间比较时,会进行值之间的比较,比如:

1 == 1 // true
1 == 2 // false
1.33 == 1.33 // true

诸如以上的例子。

同样的还可以进行对象之间的比较,如果是对象之间的比较的话,则会比较两个变量所指向对象在内存中的地址,也就是说如果两个变量没有指向同一个对象的话,得到的就是 false;

null == Integer.valueOf(1) // false
new Integer(1) == Integer.valueOf(1) // false

Integer val1 = new Integer(13);
Integer val2 = new Integer(13);
val1 == val2; // false

这里不对 ==equals的区别做介绍,如果想要了解的可以自行查阅。

我想详细描述的是我遇到的一种情况,是引用数据类型与基本数据类型之间用==比较的话会发生什么。

因为我的印象中 == 是不会引起空指针异常的,顶多一方为 null 而另外一方有值时会返回 false
但是在这种情况在引用数据类型与基本数据类型进行比较的时候发生了。

null == 1 // NullPointerException

正常的情况来讲,当引用数据类型与基本数据类型进行比较的时候,会将引用数据类型一方先进行拆箱操作(unbox),然后对两方进行值比较:

1 == Integer.valueOf(1); // true
1 == new Integer(1); // true

但是如果传入的变量是一个 null的话,就会导致拆箱操作无法正常进行,从而导致抛出一个 NullPointerException

由于拆箱操作是隐式进行的,对于开发者而言如果不知道发生了拆箱操作的话,就很难定位到空指针的位置,因此在进行等值判断的时候,建议尽量使用jdk自带的工具方法:

Objects.equals(null,1); // false

而它内部的实现是这样子的:

public static boolean equals(Object a, Object b) {
    return (a == b) || (a != null && a.equals(b));
}

对于引用数据类型和基本数据类型的比较,它首先会将传入的基本数据类型进行装箱操作(box),然后进行对象之间的比较(比较地址),在不相同的情况下再通过 equals进行判断,也就是对==等值操作做了进一步的封装。
在这里插入图片描述

总结

  • 如果是基本数据类型,==判断的是值
  • 如果是对象类型,==判断的是对象的地址
  • 如果一边是基本数据类型,另一边是对象类型,则会首先对对象类型进行拆箱,然后按照基本数据类型来处理。

我需要画重点的地方是 == 有可能会引起拆箱操作,当传入对象为 null时拆箱操作会引发空指针异常问题。

建议在使用 == 的场景下统一使用 Objects.equals来代替。

参考资料

Java中equals和==的区别
Java中的==

  Java知识库 最新文章
计算距离春节还有多长时间
系统开发系列 之WebService(spring框架+ma
springBoot+Cache(自定义有效时间配置)
SpringBoot整合mybatis实现增删改查、分页查
spring教程
SpringBoot+Vue实现美食交流网站的设计与实
虚拟机内存结构以及虚拟机中销毁和新建对象
SpringMVC---原理
小李同学: Java如何按多个字段分组
打印票据--java
上一篇文章      下一篇文章      查看所有文章
加:2021-09-27 13:57:03  更:2021-09-27 13:57:27 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/23 19:25:42-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码