为什么如果重写了某个类的equals方法,还必须重写对应的hashcode方法?

为什么如果重写了某个类的equals方法,还必须重写对应的hashcode方法?

答:

  1. 保证equals相同的两个对象hashcode必须相同的原则。
  2. 不重写hashcode方法的的话,若用hashmap/hashset等散列表存储这个类,可能会出现两个相同对象hashcode却不同,导致判定为不同元素的问题。

第一个原因可以认为是一个大原则,第二个原因则可以认为是具体表现,下面具体解释第二个原因。

详细解释

比如hashmap插入时,会先判定新元素的hash值和旧元素的hash值是否相等,hash值相等前提下,再判断equals方法。如果不重写hashcode,则equal相等的两个元素,对应的hashcode可能会不同,就会导致本来两个相等的元素被判定为不同元素,重复插入。(所以一定要保证equals相同时,hashcode也相同,但反过来不用,因为允许冲突嘛)

实验验证

创建一个people对象,如果两个people对象的name以及age相同,就认为是相同的对象,要保证相同的people不能重复加入到hashset/hashmap中,如何实现?

做法一:定义people对象,只重写equals方法

java 复制代码
class people{
    String name;
    int age;

    public people(String name, int age) {
        this.name = name;
        this.age = age;
    }

    @Override
    public String toString() {
        return "people{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        people people = (people) o;
        return age == people.age &&
                Objects.equals(name, people.name);
    }
}

测试代码

java 复制代码
public class HashSet_test1 {
    public static void main(String[] args) {
        HashSet s = new HashSet();
        s.add(new people("张三",20));
        s.add(new people("张三",20));
        s.add(new people("张三",21));
        System.out.println(s);
    }
}

测试结果,set里面插入了两个相同的对象,有问题

做法2:重写equals方法以及hashcode方法,保证name和age相同时,hashcode也相同

java 复制代码
    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        people people = (people) o;
        return age == people.age &&
                Objects.equals(name, people.name);
    }
    @Override
    public int hashCode() {
        return Objects.hash(name, age);//使用Objects工具类提供的hash方法
    }

测试结果,set插入正常

结论:重写equals方法后,hashcode方法也必须对应重写

相关推荐
mumu_wangwei4 小时前
【QFS】Golang自研的QFS分布式文件系统,QFS文件系统使用
开发语言·后端·golang
兰令水4 小时前
topcode【随机算法题】【2026.5.24打卡-java版本】
java·开发语言·算法
LCG元4 小时前
Istio - 服务网格流量治理深度解析:灰度发布 / 故障注入配置实践
java·数据库·istio
hef2884 小时前
Java Switch和Break语句用法详解:从入门到实战
java·开发语言
ABCDEEE74 小时前
3.RAG
java·linux·服务器
techdashen4 小时前
在 Rust 异步接口的丛林中生存:从同步 I/O 到手写异步状态机
开发语言·后端·rust
SuniaWang4 小时前
《Agentx专栏》03-架构设计:AgentX的六层架构是如何生长出来的
java·数据库·redis·docker·ai·架构
EnCi Zheng4 小时前
09a-斯坦福 CS336 作业一:BPE分词器
开发语言·python·算法
Refrain_zc4 小时前
Android开发在线音频播放器之章节一 AudioPlayerManager
java
Refrain_zc4 小时前
Android开发Room数据库使用(可复制)
java