当前位置: 首页 > 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()的情况


文章转载自:
http://ahriman.lbooon.cn
http://chemotactic.lbooon.cn
http://bothnia.lbooon.cn
http://bazookaman.lbooon.cn
http://aruspex.lbooon.cn
http://buckingham.lbooon.cn
http://baee.lbooon.cn
http://bilateral.lbooon.cn
http://arrack.lbooon.cn
http://bushwa.lbooon.cn
http://ataractic.lbooon.cn
http://absurdness.lbooon.cn
http://bella.lbooon.cn
http://barbiturism.lbooon.cn
http://boots.lbooon.cn
http://calces.lbooon.cn
http://camphire.lbooon.cn
http://astuteness.lbooon.cn
http://carifta.lbooon.cn
http://biomagnify.lbooon.cn
http://anarchism.lbooon.cn
http://begem.lbooon.cn
http://antipodes.lbooon.cn
http://cathomycin.lbooon.cn
http://assort.lbooon.cn
http://carthage.lbooon.cn
http://accessional.lbooon.cn
http://aoc.lbooon.cn
http://audiotactile.lbooon.cn
http://cerebrotonic.lbooon.cn
http://www.dtcms.com/a/187041.html

相关文章:

  • 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 系统日志工具详解
  • c++STL-vector的模拟实现
  • 开关电源滤波器讲解
  • 鲁滨逊归结原理详解:期末考点+解题指南
  • c++刷题便捷函数(类似于stoi的小函数)
  • 解锁性能密码:Linux 环境下 Oracle 大页配置全攻略​
  • 基于大模型的甲状腺结节诊疗全流程预测与方案研究报告
  • 基于STM32、HAL库的RN7302电能计量芯片驱动程序设计
  • Docke容器下JAVA系统时间与Linux服务器时间不一致问题解决办法
  • latex控制表格宽度,不要超出页面
  • java 中 pojo 的详细讲解