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

【HashMap】为什么用自定义的类做HashMap的Key时需要重写hashcode方法和equals方法

【HashMap】为什么用自定义的类做HashMap的Key时需要重写hashcode方法和equals方法

  • 【一】为什么有这个问题
  • 【二】Object类的中的hashcode方法和equals方法
  • 【三】重写hashcode
  • 【四】重写equals方法
  • 【五】hashmap中使用hashcode和equals方法

【一】为什么有这个问题

因为HashMap的key有个特征,就是key值不能重复,否则add的时候会覆盖。那么如果使用自定义的类做HashMap的key,就需要用到equals方法判断两个对象是否相等,判断的条件就是hashCode生成的哈希值。

Java中所有的类都是Object类的子类,所以自定义的类也是Object类的子类,Object中自带的hashcode方法和equals方法,如果不重写hashcode方法和equals方法,即使两个自定义类的对象内容完全相同,也不会判定为重复,这也就违背了HashMap的key不可重复的特点。

【二】Object类的中的hashcode方法和equals方法

Java中所有的类都是Object类的子类,Object中自带的hashcode方法,就是通过计算内存地址转换一个整数实现的,是一个native方法,返回值是int类型:

public native int hashCode();

Object中自带的equals方法,从源码中可以看出,就是将比较与被比较对象用==进行比较,用的还是比较hash值的方式

public boolean equals(Object obj) {return (this == obj);
}

【三】重写hashcode

如果不重写hashcode方法,那么即便是两个内容完全相同key对象,每次通过hashcode方法计算出的hash值也都是不一样的,因为是根据其对象所在内存地址计算的,因而做存储时也会被当作时不同的key;

重写hashcode方法该怎么写呢,这没有固定的格式,项目中看到很多中写法,列举两个

(1)如果是一个实体类,有唯一的字段id,那么可以利用该字段直接作为其hash值,如下

public class user{private int id;private String name;@overridepublic int hashcode(){return this.id;}
}

(2)也可以是如下写法,因为String类中已经重写了hashcode方法,所以当name值相同时,其对应hash值一定相同

public class User{private int id;private String name;@overridepublic int hashcode(){return id*name.hashcode;}
}

【四】重写equals方法

equals方法在上面已经介绍过了,其实就是用==进行比较,如果不重写那么即便是两个内容完全相同的对象,equals方法返回值依然是false,和hashcode类似也需要进行类似的方法重写:

(1)思路是先使用instanceof关键字判断这个被比较对象与对象是否属于一个类,返回值是true的话,再将二者转换成json字符串,最后通过String已经重写好了的equals方法进行比较,如下:

public class User{private int id;private String name;@overridepublic boolean equals(Object object){if(object instanceof User){User compareObj = (User)object;String soureStr = JsonUtils.objectToJson(this);String targetStr = JsonUtils.objectToJson(compareObj);return soureStr.equals(targetStr);}return false;}
}

【五】hashmap中使用hashcode和equals方法

举例,将一个对象user对象作为key存入hashmap当中,会经历以下流程,能够看出在判断两个对象是否相等时,要经过以下步骤:

(1)是先去判断二者的hash值是否相同,因为计算和比较hash值所耗费代价要远低于通过equals进行比较,所以先比较hash值也可以提高效率;
(2)如果比较结果不相等,则直接返回false;
(3)如果比较结果相等,则使用equals方法进行比较;
(4)如果相等,则返回true,如果不相等,则返回fasle。

在这里插入图片描述

总结:在hashmap中,对于两个对象的比较,都充分用到了key对象的hashcode方法和equals方法;

如果不重写hashcode方法,那么即便是两个内容完全一致的key对象,它仍会视为是不同的key,导致数据存取无法进行;

如果不重写equals方法,那么equals方法只是简单的将两个对象进行==比较,实际上也是通过对象间的内存地址进行比较,也导致上述情况的发生。

http://www.lryc.cn/news/92732.html

相关文章:

  • Flutter自定义对话框返回相关问题汇总
  • 002docker 安装
  • 软件工程师,全面思考问题很重要
  • 1.Apollo部署-linux
  • 【HTML】form标签
  • 基于SPAD / SiPM技术的激光雷达方案
  • 使用MATLAB工具模拟单/双频GPS和载波相位差分GPS
  • 当社恐成为技术面试官
  • Jetpack Compose:使用PagerIndicator和Infinity实现滚动的HorizontalPager
  • 2023年杭州/广州/东莞/深圳软考(中/高级)认证,618报名特惠
  • springboot项目外卖管理 day03-公共字段填充与新增删除分类
  • Nginx:Tomcat部署及优化(一)
  • Docker Swarm 集群搭建和使用 —— 筑梦之路
  • 是否需要更换CRM系统如何评估?如何确保更换成功?
  • CSDN竞赛57期题解
  • springboot+vue.js大学生竞赛报名作品评分管理系统
  • Python爱好者的自我修养(1):简单输入与输出
  • java SSM 摄影作品网站myeclipse开发mysql数据库springMVC模式java编程计算机网页设计
  • [Maven高级]->近万字文章带你深入了解Maven
  • 物联网Lora模块从入门到精通(五)光照与温湿度传感器
  • 【网络编程】计算机网络基础知识总结 | 运输层 |TCP协议
  • python关键知识点
  • c# 从零到精通 数组的操作-将两个一维数组合并成一个二维数组
  • Linux目录结构(与window目录结构对比+绝对路径和相对路径)
  • 投票活动小程序开发搭建
  • 代码随想录day18
  • QT+OpenGL高级光照 Blinn-Phong和Gamma校正
  • 【Ubuntu系统内核更新与卸载】
  • RL - 强化学习 马尔可夫奖励过程 (MRP) 的状态价值
  • Mybatis之批处理流式查询