概念:hashCode()方法和equals()方法的作用其实一样,在Java里都是用来对比两个对象是否相等一致。
hashCode() 的作用是获取哈希码(也称为散列码);它实际上是返回一个int整数。这个**哈希码的作用是确定该对象在哈希表中的索引位置。**hashCode() 定义在JDK的Object.java中,Java中的任何类都包含有hashCode() 函数。
哈希表(散列表)存储的是键值对(key-value),它的特点是:能根据“键”快速的检索出对应的“值”。这其中就利用到了散列码!(可以快速找到所需要的对象)。
注:hashCode()的默认行为是对堆上的对象产生独特值(默认返回当前对象的地址)。如果没有重写hashCode(),则该class的两个对象无论如何都不会相等(即使这两个对象指向相同的数据)。
区别:两者的区别主要体现在性能和可靠性。两者都实现对比,为什么要有HashCode?
- 性能:因为重写的equals()里一般比较的比较全面比较复杂,这样效率就比较低,而利用hashCode()进行对比,则只要生成一个hash值进行比较就可以了,效率很高。
以“HashSet如何检查重复”为例子来说明为什么要有hashCode:
对象加入HashSet时,HashSet会先计算对象的hashcode值来判断对象加入的位置,看该位置是否有值,如果没有、HashSet会假设对象没有重复出现。但是如果发现有值,这时会再调用equals()方法来检查两个对象是否真的相同。如果两者相同,HashSet就不会让其加入操作成功。如果不同的话,就会重新散列到其他位置。这样就大大减少了equals的次数(避免新加入的对象与内部每个对象进行比较),相应就大大提高了执行速度。
- 可靠性:
- equals()相等的两个对象他们的hashCode()肯定相等,也就是用equals()对比是绝对可靠的。
- hashCode()相等的两个对象他们的equals()不一定相等,也就是hashCode()不是绝对可靠的。
补充:
阿里开发规范:
- 只要重写 equals,就必须重写 hashCode;
- 因为 Set 存储的是不重复的对象(上例),依据 hashCode 和 equals 进行判断,所以 Set 存储的对象必须重写这两个方法;
- 如果自定义对象做为 Map 的键,那么必须重写 hashCode 和 equals;
- String 重写了 hashCode 和 equals 方法,所以我们可以非常愉快地使用 String 对象作为 key 来使用;
什么时候重写hashCode?
- 一般的地方不需要,只有当类需要放在HashTable、HashMap、HashSet等等hash结构的集合时才会重写hashCode。
为什么要重写hashCode呢?
- 如果你重写了equals,比如说是基于对象的内容实现的,而保留hashCode的实现不变,那么很可能某两个对象明明是“相等”,而hashCode却不一样。这样,当你用其中的一个作为键保存到hashMap、hasoTable或hashSet中,再以“相等的”找另一个作为键值去查找他们的时候,则根本找不到。
参考:https://blog.csdn.net/xlgen157387/article/details/88087963