0

0

为什么Java中要重写hashCode_hashCode与集合结构关联的原理解析

P粉602998670

P粉602998670

发布时间:2025-11-26 08:39:26

|

703人浏览过

|

来源于php中文网

原创

重写hashCode是确保自定义对象在哈希集合中正确存储和查找的关键。Java规定:若两个对象equals相等,则其hashCode必须相同。以Person类为例,若只重写equals而未重写hashCode,尽管两对象逻辑相等,但默认hashCode基于内存地址,可能导致不同哈希值,使HashSet或HashMap将它们视为不同对象,造成重复添加或查找失败。哈希集合如HashMap通过hashCode确定对象存储桶位置,再用equals判断是否真正重复。因此,若自定义类作为键或元素且重写了equals,就必须重写hashCode,通常使用Objects.hash()基于equals涉及字段生成一致哈希值,以维护集合的唯一性和正确性。

为什么java中要重写hashcode_hashcode与集合结构关联的原理解析

在Java中,重写hashCode方法与集合(尤其是基于哈希的集合如HashMapHashSetHashtable)的行为密切相关。如果不正确地重写hashCode,会导致对象无法被正确存储和查找。核心原因在于:这些集合依赖hashCode来决定对象的存储位置。

hashCode 与 equals 的契约关系

Java官方明确规定:如果两个对象通过equals方法比较相等,那么它们的hashCode必须相同。反之则不成立——hashCode相同,对象不一定相等(可能发生哈希冲突)。

这个契约是哈希集合正常工作的基础。一旦违反,就会出现逻辑错误。

例如:

你创建了一个Person类,只重写了equals,而没有重写hashCode。两个Person对象名字相同,equals返回true,但它们的hashCode来自Object默认实现(通常是内存地址),很可能不同。当你把其中一个放入HashSet后,再用另一个相等的对象去查找,系统会根据不同的hashCode去不同的桶查找,结果找不到,导致集合“丢失”了本应存在的对象。

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

哈希集合如何使用 hashCode

HashMap为例,其内部由数组+链表/红黑树构成。当插入一个键值对时:

  • 先调用键对象的hashCode方法,得到一个整数
  • 通过哈希算法(如取模)将该整数映射到数组的某个索引位置(即“桶”)
  • 若该桶已有元素,则使用equals方法判断是否重复,避免键冲突

因此,hashCode 决定了对象存放在哪个桶,equals 决定在桶内是否真正相等

Kite
Kite

代码检测和自动完成工具

下载

为什么必须重写 hashCode

如果你自定义类作为HashMap的键或HashSet的元素,而没有重写hashCode

  • 默认的hashCode基于内存地址生成
  • 即使两个对象逻辑上相等(equalstrue),也会被分配到不同桶中
  • 导致集合误判为“新元素”,可能重复添加,或无法查找到已存在的对象

这破坏了集合应有的语义,比如Set不能保证唯一性,Map无法通过等价键取值。

正确重写 hashCode 的建议

重写时要确保:相等的对象产生相同的哈希码。常用做法是基于equals中涉及的字段计算哈希值。

例如:


public class Person {
    private String name;
    private int age;

    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (!(o instanceof Person)) return false;
        Person person = (Person) o;
        return age == person.age && Objects.equals(name, person.name);
    }

    @Override
    public int hashCode() {
        return Objects.hash(name, age);
    }
}

这样,只要name和age相同,hashCode就一致,能正确映射到同一个桶中,再通过equals确认是否真重复。

基本上就这些。不复杂但容易忽略。只要记住:用了自定义类做键或集合元素,且重写了equals,就必须重写hashCode,否则哈希集合会出问题。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

844

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

742

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

740

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

400

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

431

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16926

2023.08.03

c++空格相关教程合集
c++空格相关教程合集

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

0

2026.01.23

热门下载

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

精品课程

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

共23课时 | 2.8万人学习

C# 教程
C# 教程

共94课时 | 7.4万人学习

Java 教程
Java 教程

共578课时 | 49.9万人学习

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

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