当前位置: 首页 > news >正文

hashCode()和equals(),为什么使用Map要重写这两个,为什么重写了hashCode,equals也需要重写

文章目录

    • hashCode()
    • equals()
    • HashMap的存储规则
    • 未重写hashCode()和equals()的情况
    • 仅重写equals()的情况
    • 仅重写hashCode()的情况
    • 同时重写hashCode()和equals()的情况
    • 关键结论
    • 总结

hashCode()

默认返回对象的内存地址转换的int值。

equals()

默认比较两个对象的内存地址是否相同。

HashMap的存储规则

  • 当对象需要存入map时,使用hashCode() 获得对象的hash码,和数组的长度取余得到一个存储位置
  • 如果位置为空,直接存入
  • 如果位置不为空,说明发生了hash冲突,使用equals进行判断,是否是相同的
  • equals判断相同,则说明存储元素重复,map覆盖,set不存入
  • equals判断不相同,说明元素应该存入,散列到其他位置

未重写hashCode()和equals()的情况

  1. hashCode()
  • 默认返回对象的内存地址转换的int值
  • 相同对象(内存地址相同)返回相同的hashCode
  1. equals()
  • 默认比较两个对象的 内存地址 是否相同
  1. 存入Map的行为
    • 如果两个属性值相同的对象存入Map
      • 第一次存入时,计算hashCode并放入对应桶
      • 第二次存入时,由于内存地址不同:
        • hashCode() 可能不同(取决于 JVM 实现),可能发生哈希冲突
        • 即使 hashCode 相同, equals()比较内存地址仍返回false
        • 最终两个对象都会存入Map(违反业务逻辑)

仅重写equals()的情况

  1. hashCode()
  • 未重写,仍返回内存地址的hashCode
  1. equals()
  • 重写后比较对象的属性值是否相同
  1. 存入Map的行为
    • 如果两个属性值相同的对象存入Map
      • 第一次存入时,计算 hashCode 并放入对应桶
      • 第二次存入时:
        • hashCode() 可能不同(内存地址不同),可能发生哈希冲突
        • 即使hashCode相同,equals()比较属性值返回 true ,但Map仍认为它们是不同对象(因为内存地址不同)
        • 最终两个对象都会存入Map(违反业务逻辑)

仅重写hashCode()的情况

  1. hashCode()
  • 根据对象的属性值计算hashCode
  1. equals()
  • 默认比较两个对象的内存地址是否相同
  1. 存入Map的行为
    • 如果两个属性值相同的对象存入Map
      • 第一次存入时,计算hashCode并放入对应桶
      • 第二次存入时:
        • hashCode()相同(属性值相同),进入同一桶
        • 即使hashCode相同,但是equals()比较属性值返回false(因为内存地址不同)
        • 最终两个对象都会存入Map (违反业务逻辑)

同时重写hashCode()和equals()的情况

  1. hashCode()
  • 根据对象的属性值计算hashCode
  1. equals()
  • 比较对象的属性值是否相同
  1. 存入Map的行为
  • 如果两个属性值相同的对象存入Map
    • 第一次存入时计算hashCode并放入对应桶
    • 第二次存入时:
      • hashCode()相同(属性值相同),进入同一桶
      • equals()比较属性值返回true,Map认为是同一个对象
      • 最终只存入一次(符合业务逻辑)

关键结论

  1. hashCode()和equals()必须同时重写
  • 如果只重写 equals() , Map 可能存储重复对象(因为 hashCode() 仍比较内存地址)。
  • 如果只重写 hashCode() , equals() 仍比较内存地址,可能导致逻辑错误(如 HashSet 判断重复失效)。
  1. 两套逻辑的本质
    • 内存地址比较 (未重写):适用于判断对象是否是同一个实例。
    • 属性值比较 (重写后):适用于判断业务上是否是“相同”的对象(如两个 User 对象 id 和 name 相同即视为相同)。
  2. Map的存储规则
  • 先比较 hashCode() ,再比较 equals() 。
  • 只有 hashCode()相同且equals()返回true时,才认为是同一个对象。

总结

本质上就是使用了两套逻辑进行判断,一套使用的是地址,一套使用的是属性值,所以如果只重写一部分,就会发生问题,存储重复的值。

另外使用hashMap存储对象,必须要重写这两个方法,原因参考未重写hashCode()和equals()的情况

相关文章:

  • csdn博客打赏功能
  • 小刚说C语言刷题—1149 - 回文数个数
  • 什么是IP专线?企业数字化转型的关键网络基础设施
  • 大小端的判断方法
  • cursor对话关键词技巧
  • spring boot3.0自定义校验注解:文章状态校验示例
  • PH热榜 | 2025-05-12
  • 前端vue+elementplus实现上传通用组件
  • SHAP分析!Transformer-GRU组合模型SHAP分析,模型可解释不在发愁!
  • HDFS客户端操作
  • 排查服务器内存空间预警思路
  • AI日报 - 2024年05月13日
  • 航电系统之电传飞行控制系统篇
  • Excel VBA 与 AcroForm 文档级脚本对比
  • MCU开启浮点计算FPU
  • [springboot]SSM日期数据转换易见问题
  • Linux电源管理(五),发热管理(thermal),温度控制
  • C 语 言 - - - 简 易 通 讯 录
  • Python 字符串
  • 【Linux 系统调试】syslog:Linux 系统日志工具详解
  • 王毅集体会见加勒比建交国外长及代表
  • 夜读丨取稿费的乐趣
  • 韩国总统选战打响:7人角逐李在明领跑,执政党临阵换将陷入分裂
  • 国家统计局今年将在全国开展两次人口固定样本跟访调查
  • 泽连斯基表示将在土耳其“等候”普京
  • 著名学者黄修己去世,享年90岁