基本概念
- 要比较两个对象是否相等时需要调用对象的equals() 方法: 判断对象引用所指向的对象地址是否相等
- 对象地址相等时, 那么对象相关的数据也相等,包括: 对象句柄 对象头 对象实例数据 对象类型数据
- 可以通过比较对象的地址来判断对象是否相等
Object源码
- 对象在不重写的情况下使用的是Object中的equals() 方法和hashCode() 方法 equals(): 判断的是两个对象的引用是否指向同一个对象 hashCode(): 根据对象地址生成一个整数数值
- Object的hashCode() 方法修饰符为native: 表明该方法是由操作系统实现. Java调用操作系统底层代码获取Hash值
public native int hashCode();
重写equals
- 重写equals()方法的场景: 假设现在有很多学生对象 默认情况下,要判断多个学生对象是否相等,需要根据地址判断: 若对象地址相等,那么对象实例的数据一定是一样的 判断相等的要求: 当学生的姓名,年龄,性别相等时,认为对象是相等的, 不一定需要对象的地址完全相同
- 重写了equals() 方法后,这里会输出 [s1==s2]
- 如果没有重写 equals() 方法,那么必定会输出 [s1!=s2]
重写hashCode
- 根据重写equals的方法,上述s1和s2认为是相等的
- Object中的hashCode()方法: 在equals() 方法没被修改的前提下,多次调用同一个对象的hashCode() 方法返回的值必须是相同的正数 如果两个对象互相equals(), 那么这两个对象的hashcode值必须相等 为不同的对象生成不同的hashcode可以提升Hash表的性能
- HashSet底层实现: HashSet底层是通过HashMap实现的 比较Set容器内元素是否相等是通过比较对象的hashcode来判断是否相等的
- hashCode()的写法: 首先整理出判断对象相等的属性 然后去一个尽可能小的正整数,防止最终结果超出整型int的取数范围 然后计算[正整数 * 属性的hashCode + 其余某个属性的hashCode] 重复步骤
原理分析
- 因为没有重写父类的Object的hashCode() 方法,所以Object的hashCode() 方***根据两个对象的地址生成响应的hashcode
- 由于两个对象分别是实体类创建的不同的实例,所以地址肯定是不一样的,那么hashcode值也是不一样的
- Set区别对象是不是唯一的标准: 两个对象的hashcode值是否一样 然后再判定两个对象是否equals
- Map区别对象是不是唯一的标准: 先根据Key值的hashcode分配来获取保存数组下标 然后再根据eaquals区分是否是唯一值
HashMap
HashMap组成结构
HashMap的存储
- HashMap的存储: 一个对象存储到HashMap中的位置是由key的hashcode值决定的 HashMap查找key: 查找key时 ,hashMap会先根据key值的hashcode经过取余算法定位所在数组的位置 然后根据key的equals方法匹配相同的key值获取相应的对象
- 存值规则: 将Key的hashcode与HashMap的容量,进行取余运算得出该Key存储在数组所在位置的下标
- HashMap查找key: 得到key在数组中的位置 匹配得到对应key值对象 然后将上述多个对象根据key.equals() 来匹配获取对应的key的数据对象
- HashMap中的hashCode: 如果没有hashcode就意味着HashMap存储的时候是没有规律可循的 这样每次使用map.get() 方法,就要将map里的对象一一进行equals匹配,导致效率低下
作者:攻城狮Chova
链接:https://juejin.cn/post/6991821893711953927
来源:掘金