0

0

在Java里equals与hashCode为何成对出现_Java对象比较规则说明

P粉602998670

P粉602998670

发布时间:2026-01-13 14:54:09

|

956人浏览过

|

来源于php中文网

原创

必须重写hashCode(),因为哈希集合依赖hashCode()定位桶、equals()精确比对;若只重写equals(),相等对象可能散列到不同桶,导致HashSet重复插入或HashMap查不到key。

在java里equals与hashcode为何成对出现_java对象比较规则说明

为什么重写 equals() 就必须重写 hashCode()

因为哈希集合(如 HashSetHashMap)内部依赖「先查桶、再比值」的两层机制:先用 hashCode() 定位到数组下标(桶),再用 equals() 在桶内精确比对。如果只重写 equals() 而不重写 hashCode(),逻辑上相等的对象可能被散列到不同桶里,导致 HashSet 误存重复项、HashMap 查不到已存在的 key。

  • 默认 Object.hashCode() 返回的是对象内存地址的哈希值,两个内容相同但不同实例的对象,hashCode() 必然不同
  • HashMap.get(key) 的源码中,第一道判断就是 e.hash == hash;不通过就直接跳过后续 equals() 比较
  • 违反契约(equals() 相等 ⇒ hashCode() 必须相等)会导致集合行为不可预测,且这种 bug 很难通过单元测试覆盖到

不配对重写的典型故障现象

最常见的是「明明 equals() 返回 true,但 HashSet 还是加进去了两个对象」或「HashMap 里存了 key,却 get() 不出来」。

  • HashSet.add(new Person("Alice", 25))HashSet.add(new Person("Alice", 25)) —— 如果没重写 hashCode(),size 可能是 2 而不是 1
  • map.put(new Apple("red"), 10) 后,map.get(new Apple("red")) 返回 null,哪怕 Apple.equals() 已正确返回 true
  • 调试时发现两个对象 equals()true,但它们的 hashCode() 值差得离谱(比如 -129384723 vs 192837465

正确重写姿势:3 条硬约束 + 1 个推荐工具

核心不是“怎么写”,而是“不能漏什么”。Java 规范只要求满足契约,不规定实现方式,但实际开发中必须守住底线:

灵机语音
灵机语音

灵机语音

下载
  • 所有参与 equals() 判等的字段,也必须参与 hashCode() 计算(反之不一定)
  • hashCode() 计算中避免使用可变字段(如未声明 final 且后续会修改的属性),否则对象加入 HashSet 后再改字段,就再也找不到了
  • 空值处理要一致:equals() 中用 Objects.equals(a, b)hashCode() 中就用 Objects.hash(a, b)
  • 强烈建议用 IDE 自动生成(IntelliJ → Alt+Insert → “Generate” → 选 equals()hashCode()),它会自动处理 null、类型检查、字段一致性
public class User {
    private final String name;
    private final int age;

    // 构造、getter 略

    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        User user = (User) o;
        return age == user.age && Objects.equals(name, user.name);
    }

    @Override
    public int hashCode() {
        return Objects.hash(name, age); // ← 与 equals 中的字段严格一致
    }
}

容易被忽略的坑:继承场景下的陷阱

子类如果扩展了父类的判等逻辑(比如新增字段),但没重写 equals()hashCode(),或者只重写了其中一个,问题会更隐蔽——尤其当父类已重写过这对方法时。

立即学习Java免费学习笔记(深入)”;

  • 子类重写 equals() 时,必须调用 super.equals(),否则可能破坏对称性(a.equals(b)true,但 b.equals(a)false
  • 子类的 hashCode() 必须包含父类参与判等的所有字段,否则父子实例之间可能违反「相等必同哈希」契约
  • 若父类没重写 hashCode(),子类重写 equals() 却忘了重写 hashCode(),那子类实例的哈希值仍来自 Object 默认实现,直接失效
真正麻烦的从来不是“怎么写”,而是“什么时候该写”——只要你的类会被放进任何基于哈希的集合,或者作为 HashMap 的 key,就必须成对重写。漏掉一次,就可能埋下一个线上查不出来源的 null 返回或重复插入。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

254

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

1111

2024.03.01

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

254

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

1111

2024.03.01

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

254

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

1111

2024.03.01

golang map内存释放
golang map内存释放

本专题整合了golang map内存相关教程,阅读专题下面的文章了解更多相关内容。

77

2025.09.05

golang map相关教程
golang map相关教程

本专题整合了golang map相关教程,阅读专题下面的文章了解更多详细内容。

40

2025.11.16

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

69

2026.03.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.4万人学习

C# 教程
C# 教程

共94课时 | 11.4万人学习

Java 教程
Java 教程

共578课时 | 82.5万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号