Java 包装类型 == 操作引发的 Bug

背景

一个简单的列表检索功能,列表元素有一个 Long 类型的属性,遍历过程中犯了一个低级错误,导致功能流程始终错误,本文将分享两个低级错误引发的 bug。

两个 Long 类型的 ==

查找某个元素 A 在列表 B 中对应的对象的时候,根据元素主键查询,主键类型为包装类型 Long ,遍历流程如下:

复制代码
for(MyData temp:b){
     if (temp.getId() == a.getId() { //  MyData 的 id 属性为 Long 类型
          return temp;
     }
}

这么一段简单的查找代码,结果怎么都找不到目标对象,断点调试发现问题出在 == 操作上,改成 equals 就可以了。

关于 Java Long 的包装类型和元素类型的判断相等的操作回顾:

复制代码
Long a = 81487354807713792L;
Long b = 81487354807713792L;
System.out.println(a==b); // false
System.out.println(a.equals(b)); // true

long c = 81487354807713792L;
long d = 81487354807713792L;
System.out.println(c==d); // true

对 Collections.EMPTY_SET 进行 add 引发的异常

另一个低级错误是对 Collections.EMPTY_SET 进行 add 引发的,需要合并两个集合,第一个集合 A 可能是 Collections.EMPTY_SET ,最终将另一个集合 B 合并到 A 得到一个大集合。

复制代码
Set<MyData> a = getDatas();// 如果为空,返回了 Collections.EMPTY_SET
Set<MyData> b = getDatas1();
a.addAll(b);

当集合 a 为集合的空对象时,操作异常:

复制代码
Exception in thread "main" java.lang.UnsupportedOperationException
	at java.util.AbstractCollection.add(AbstractCollection.java:262)

修正方式 :如果需要直接对一个集合进行 add ,就不能用 Collections.EMPTY_SET

Collections 的空集合使用注意事项

Collections.EMPTY_SET 为例,跟源码它的定义主要是下面三行代码:

复制代码
public static final Set EMPTY_SET = new EmptySet<>();

private static class EmptySet<E> extends AbstractSet<E>

AbstractSet<E> extends AbstractCollection

核心在于 AbstractCollection 类的 add,默认直接抛出了异常,限制了空集合不允许添加:

复制代码
public boolean add(E e) {
     throw new UnsupportedOperationException();
 }

结论java.util.Collections 类中所有的 EMPTY_XXX 对象都不能进行 add 操作。

启示录

定位到这两个低级错误后,想起那句调侃:代码编写分分钟,bug 查找两小时。这两个问题恰好是一个比较复杂的流程的一部分,构建环境进行测试,测一次差不多十几分钟,加上机器怠工,跟这俩小问题,耗了两个小时。

以此文为记,这么深刻的过程,估计以后也不会再犯了!

相关推荐
月亮不月亮9 分钟前
月亮商场购物打折Java
java·eclipse
guozhetao19 分钟前
【ST表、倍增】P7167 [eJOI 2020] Fountain (Day1)
java·c++·python·算法·leetcode·深度优先·图论
技术思考者20 分钟前
基础很薄弱如何规划考研
java·经验分享·考研
●VON43 分钟前
重生之我在暑假学习微服务第二天《MybatisPlus-下篇》
java·学习·微服务·架构·mybatis-plus
老华带你飞43 分钟前
口腔助手|口腔挂号预约小程序|基于微信小程序的口腔门诊预约系统的设计与实现(源码+数据库+文档)
java·数据库·微信小程序·小程序·论文·毕设·口腔小程序
枫叶丹41 小时前
【Qt开发】信号与槽(二)-> 信号和槽的使用
开发语言·qt
hqxstudying1 小时前
J2EE模式---服务层模式
java·数据库·后端·spring·oracle·java-ee
GM_8281 小时前
【最新最完整】SpringAI-1.0.0开发MCP Server,搭建MCP Client 实战笔记(进阶+详细+完整代码)
java·后端·ai编程·springai·mcp
都叫我大帅哥1 小时前
Java DelayQueue:时间管理大师的终极武器
java
秋千码途1 小时前
小架构step系列27:Hibernate提供的validator
java·spring·架构·hibernate