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进行判断,也就是对==等值操作做了进一步的封装。

参考资料

Java中equals和==的区别

Java中的==

总结

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

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

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

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注我们的更多内容!

(0)

相关推荐

  • Java中的== 和equals()方法详解与实例

    Java中的== 和equals()方法: Java中的数据类型,可分为两类: 1.基本数据类型,也称原始数据类型. byte,short,char,int,long,float,double,boolean,他们之间的比较,应用双等号(==),比较的是他们的值. 2.引用数据类型(类) 当它们用(==)进行比较的时候,比较的是他们在内存中的存放地址,所以,除非是同一个new出来的对象,他们的比较后的结果为true,否则比较后结果为false. Java当中所有的类都是继承于Object这个基类

  • Java中的equsals和==

    目录 Java的equsals和== 1.Java 中的== 2.Java 中equals方法 Java的equsals和== 前言:在我们常用的类中equals被重写后,作用就是为了比较对象的内容,==是比较对象的内存地址.但并不能说所有的equals方法就是比较对象的内容. 1.Java 中的== 1).对于对象引用类型:"=="比较的是对象的内存地址. 比如说: String s1 = "Hello"; String s2 = new String (&quo

  • 详解 Java 中 equals 和 == 的区别

    详解 Java 中 equals 和 == 的区别 1 前言 在 Java 语言中,equals 和 == 都是用来检测两个字符串是否相等,返回值也都是布尔型(boolean),但是两者在内部比较的处理中却不尽相同,因此在需要检测两个字符串是否相等的时候,我们一定要特别的注意,选择适当的检测方式,防止造成不必要的 bug.从表面上来看,这种 bug 很像随机产生的间歇性错误. 2 区别 在需要检测两个字符串是否相等的时候,我们可以使用 equals 方法.对于表达式: s.equals(t) 如

  • Java中的==使用方法详解

    目录 Java中的== 2 字符串常量池 3 总结 Java中的== Java中的==表示的是什么呢?有时候很令人费解.比如,以下例子输出是什么? // 例一 String str = "wo"; String str1 = "wo"; System.out.println("例一:" + (str == str1)); // 例二 String str2 = new String("wo"); String str3 = n

  • Java中==与equals的区别小结

    public class Compare { /** * == 与 equals()区别 */ public static void main(String[] args) { String s1 = new String("Hello,World!"); //创建两个String型对象引用; String s2 = new String("Hello,World!"); String s3 = s1; //将s1对象引用赋予s3 System.out.printl

  • 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>

  • 阿里Druid数据连接池引发的线上异常解决

    目录 前言 过程一:定位工作流 过程二:定位JPA的OpenEntityManagerInViewInterceptor 过程三:定位Druid,真正的罪魁祸首 后记: 前言 事件起因:项目使用了activiti工作流,系统是由老的spring mvc项目改造成的spring boot项目,数据库链接池从dbcp切换到druid,新系统上线后,同事多次系统隔一段时间后数据查询就很慢,基本出不来. 由此开始了线上bug排查之路.这个问题从一开始就模糊定位到数据库层面的问题,因为只有和数据相关的操作

  • 基于java涉及父子类的异常详解

    java中的异常涉及到父子类的问题,可以归纳为一句话:子类的构造函数抛出的异常必须包含父类的异常,子类的方法可以选择抛出"范围小于等于"父类的异常或不抛出异常. 1. 为什么构造函数必须抛出包含父类的异常? 在<thingking in java>中有这么一段话: 异常限制:当覆盖方法时,只能抛出在基类方法的异常说明中列出的那些异常 异常限制对构造器不起作用,你会发现StormyInning的构造器可以抛出任何异常,而不必理会基类构造函数所抛出的异常.然而因为必须构造函数必

  • Java中正则表达式的使用和详解(上)

    1.匹配验证-验证Email是否正确 public static void main(String[] args) { // 要验证的字符串 String str = "service@xsoftlab.net"; // 邮箱验证规则 String regEx = "[a-zA-Z_]{1,}[0-9]{0,}@(([a-zA-z0-9]-*){1,}\\.){1,3}[a-zA-z\\-]{1,}"; // 编译正则表达式 Pattern pattern = Pa

  • Java大文件上传详解及实例代码

    Java大文件上传详解 前言: 上周遇到这样一个问题,客户上传高清视频(1G以上)的时候上传失败. 一开始以为是session过期或者文件大小受系统限制,导致的错误.查看了系统的配置文件没有看到文件大小限制,web.xml中seesiontimeout是30,我把它改成了120.但还是不行,有时候10分钟就崩了. 同事说,可能是客户这里服务器网络波动导致网络连接断开,我觉得有点道理.但是我在本地测试的时候发觉上传也失败,网络原因排除. 看了日志,错误为: java.lang.OutOfMemor

  • Java多线程中ReentrantLock与Condition详解

    一.ReentrantLock类 1.1什么是reentrantlock java.util.concurrent.lock中的Lock框架是锁定的一个抽象,它允许把锁定的实现作为Java类,而不是作为语言的特性来实现.这就为Lock的多种实现留下了空间,各种实现可能有不同的调度算法.性能特性或者锁定语义.ReentrantLock类实现了Lock,它拥有与synchronized相同的并发性和内存语义,但是添加了类似锁投票.定时锁等候和可中断锁等候的一些特性.此外,它还提供了在激烈争用情况下更

  • java之assert关键字用法案例详解

    Java2在1.4中新增了一个关键字:assert.在程序开发过程中使用它创建一个断言(assertion).,它的语法形式有如下所示的两种形式: 1.assert condition; 这里condition是一个必须为真(true)的表达式.如果表达式的结果为true,那么断言为真,并且无任何行动 如果表达式为false,则断言失败,则会抛出一个AssertionError对象.这个AssertionError继承于Error对象, 而Error继承于Throwable,Error是和Exc

  • Java实现图片合成的示例详解

    目录 场景 环境 搭建 引入pom文件 定义核心接口ImageService 定义核心接口实现类ImageServiceImpl 测试ImageController 测试效果 总结 场景 前端有一个神器——canvas,这个画布标签可以处理各种图片的合成,可以精确到图片的具体坐标,加水印,去水印,简直不要太简单!那java后端可以处理吗?请大声的告诉他,能,必须能!今天小编告诉你一个神器——image-combiner,合成图片so easy! 环境 jdk1.8 spring boot 搭建

  • Java Redis配置Redisson的方法详解

    目录 需要的Maven application-redis.yml Session共享配置 Redisson配置 其他Redisson的Config配置方式 需要的Maven <!--redis--> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-redis</artifactId> <e

  • java操作mongoDB查询的实例详解

    java操作mongo查询的实例详解 前言: MongoDB是一个基于分布式文件存储的数据库.由C++语言编写.旨在为WEB应用提供可扩展的高性能数据存储解决方案. MongoDB是一个介于关系数据库和非关系数据库之间的产品,是非关系数据库当中功能最丰富,最像关系数据库的.他支持的数据结构非常松散,是类似json的bson格式,因此可以存储比较复杂的数据类型.Mongo最大的特点是他支持的查询语言非常强大,其语法有点类似于面向对象的查询语言,几乎可以实现类似关系数据库单表查询的绝大部分功能,而且

随机推荐