java 为什么重写equals一定要重写hashcode?

举报
经典鸡翅 发表于 2022/02/18 00:58:13 2022/02/18
【摘要】 前言 最近复习,又看到了这个问题,在此记录和整理,通过例子来说明这种情况的原因,使大家可以清晰明白这个问题。 初步探索 首先我们要了解equals方法是什么,hashcode方法是什么。 equals方法 equals 是java的obejct类的一个方法,equals的源码如下: public boole...

前言

最近复习,又看到了这个问题,在此记录和整理,通过例子来说明这种情况的原因,使大家可以清晰明白这个问题。

初步探索

首先我们要了解equals方法是什么,hashcode方法是什么。

equals方法

equals 是java的obejct类的一个方法,equals的源码如下:


   
  1. public boolean equals(Object paramObject){
  2. return(this == paramObject);
  3. }

由此我们可以看到equals是用来比较两个对象的内存地址是否相等。

hashCode方法

hashCode方法是本地方法,用于计算出对象的一个散列值,用于判断在集合中对象是否重复的关键。

一条定理

equals相同的对象,hashCode必然相同。

代码示例

建立一个Student类。


   
  1. public class Student {
  2. private String name;
  3. private int age;
  4. private String QQ;
  5. @Override
  6. public boolean equals(Object o) {
  7. if (this == o) return true;
  8. if (o == null || getClass() != o.getClass()) return false;
  9. Student student = (Student) o;
  10. return age == student.age &&
  11. Objects.equals(name, student.name) &&
  12. Objects.equals(QQ, student.QQ);
  13. }
  14. }

在 student 类中,我们重写了equals方法。

书写一个测试类


   
  1. public class Test {
  2. public static void main(String[] args) {
  3. Student student = new Student();
  4. Student student2 = new Student();
  5. System.out.println(student.equals(student2)); //true
  6. System.out.println(student.hashCode()); //356573597
  7. System.out.println(student2.hashCode()); //1735600054
  8. HashMap<Student, String> map = new HashMap<>();
  9. map.put(student,"123");
  10. map.put(student2,"456");
  11. System.out.println(map.get(student));
  12. System.out.println(map.get(student2));
  13. }
  14. }

输出


   
  1. true
  2. 356573597 student 的hashcode值
  3. 1735600054 student 2的hashcode值
  4. 123
  5. 456

此时,我们发现 equals 相等的对象,hashcode却不相等,同时在map中用不同的对象进行了存储,map计算出来的hash值不同,但equals却相同。这时候懵了。到底两个对象一样不一样呢。
所以我们在重写equals的时候,必须重写hashcode。

重新定义 student 类


   
  1. public class Student {
  2. private String name;
  3. private int age;
  4. private String QQ;
  5. @Override
  6. public boolean equals(Object o) {
  7. if (this == o) return true;
  8. if (o == null || getClass() != o.getClass()) return false;
  9. Student student = (Student) o;
  10. return age == student.age &&
  11. Objects.equals(name, student.name) &&
  12. Objects.equals(QQ, student.QQ);
  13. }
  14. @Override
  15. public int hashCode() {
  16. return Objects.hash(name, age, QQ);
  17. }
  18. }

再次测试


   
  1. public class Test {
  2. public static void main(String[] args) {
  3. Student student = new Student();
  4. Student student2 = new Student();
  5. System.out.println(student.equals(student2)); //true
  6. System.out.println(student.hashCode()); // 29791
  7. System.out.println(student2.hashCode()); // 29791
  8. HashMap<Student, String> map = new HashMap<>();
  9. map.put(student,"123");
  10. map.put(student2,"456");
  11. System.out.println(map.get(student)); //456
  12. System.out.println(map.get(student2)); //456
  13. }
  14. }

最后的输出


   
  1. true
  2. 29791 //相同的对象
  3. 29791
  4. 456 //说明以一个值key存储,相同的值
  5. 456

几条定理

1、两个对象的equals相等,hashcode必然相等。
2、两个对象不等,hashcode也可能相等。
3、hashcode相等,对象不一定相等。
4、hashcode不等,对象一定不等。

文章来源: blog.csdn.net,作者:经典鸡翅,版权归原作者所有,如需转载,请联系作者。

原文链接:blog.csdn.net/hanqing456/article/details/111878961

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。