Bootstrap

Java中的hashCode()和equals()方法,它们与 == 操作符有什么区别?

目录

1. 三者的关系

2. 重写equals()方法时的注意事项

3. 两个方法的联系

4. 二者之间的合约

5. 应用场景

6. 通过场景来理解这两个方法


        在Java编程中,hashCode()equals()方法的重要性不言而喻,尤其是在处理集合框架时。这两个方法不仅关系到对象的比较和哈希表的效率,还涉及到对象的唯一性和存储。

1. 三者的关系

它们是用于比较对象的三种方式,但它们的用途和实现还是有挺大区别的。

  • hashCode:用于散列存储结构当中来确定对象的存储位置。可以用于快速比较两个对象是否不同,这是因为它们的哈希码不同,那么它们肯定不相等。

  • equals:用于比较两个对象的内容是否相等,这通常需要重写自定义比较逻辑。

  • == :用于比较两个引用是否指向同一个对象,即内存地址。对于基本数据类型的话,用于比较它们的值。

首先,hashCode()方法用于计算对象的哈希码,这个哈希码是一个整数值,通常用来快速确定对象在哈希表等数据结构中的存储位置。它有助于提高查找和插入的效率。哈希码的计算通常基于对象的属性,而不同的对象可能产生相同的哈希码,这也就是所谓的哈希冲突。

而equals()方法用于比较两个对象是否在逻辑上相等,默认情况下,它使用的是对象的引用地址进行比较,但在很多情况下,我们希望根据对象的属性来判断它们是否相等,这时就需要重写equals()方法,以实现自定义的相等比较逻辑。

2. 重写equals()方法时的注意事项

那么在重写equals()方法时,有哪些注意事项呢?

需要遵循五个原则:自反性、对称性、传递性、一致性、对null的比较。

3. 两个方法的联系

那么,哈希码和equals()方法有什么关联呢?

二者之间有一个最重要的关联就是在重写equals()方法时,通常也需要重写hashCode()方法,以保证相等的对象具有相同的哈希码。这是因为在使用哈希表等数据结构时,会先根据哈希码来确定存储位置,然后再使用equals()方法来处理可能的哈希冲突。用equals()判等的两个对象的哈希码一定相等,用equals()判不等的两个对象的哈希码一定不相等。

4. 二者之间的合约

在Java中,hashCode方法和equals方法之间有一个“合约”:

  • 如果两个对象根据equals方法被认为是相等的,那么它们必须具有相同的哈希码;

  • 如果两个对象具有相同的哈希码,它们并不一定相等,但会被放在同一个哈希桶中。

5. 应用场景

那在实际的工作当中,在什么情况下会遇到需要重写这些方法的场景呢?

当我们需要在自定义对象之间进行比较或者将它们存储在哈希集合中时,就需要重写这些方法。比如在实现缓存、数据唯一性校验或者自定义数据结构时,重写这些方法能够帮助我们更准确地控制对象的比较和存储行为。

6. 通过场景来理解这两个方法

哈希码的作用就是确定对象在哈希表中的索引位置。假设现在有这样的一个需求:我想让一批对象能够存储起来,不允许存储重复的对象,并且能够随时获取到对象。

一说起存储,那自然就会想到数组,我可以将对象挨个放在数组当中,当判断对象是否重复存储时,或者获取指定对象时,我们每次都得从头开始遍历整个数组,挨个儿和数组中的对象进行equals()比较,equals()结果为真,则表示找到了指定对象。这样的话确实是满足了需求,但有一个问题:就是效率太低了,每次都得遍历整个数组,假设数组有一万个对象,那每次操作时都需要比较一万次,此时的时间复杂度为O(n)。

这时,通过hashCode()获取到的哈希码就派上了用场:我们在存放对象时,可以通过哈希码来和数组的长度取余,这样就能得到数组要存放的位置。比如:数组长度为10,对象的哈希码为17,那进行取余之后得7,就可以将该对象存放到下标为7的位置上,这样无论是存储元素,还是获取元素,通过数组下标就只需要操作一次,此时的时间复杂度为O(1)。

那么哈希码的作用就是:确定索引位置,就能大幅提升效率。但是接下来呢,又会有一个很大的问题:那就是哈希码是可以重复的,毕竟哈希码只是通过一定的逻辑,计算出来的int数值,那两个不同的对象,完全有可能哈希码会相同,这也就是我们常说的哈希冲突。

当要存储的对象和已经存储的对象发生哈希冲突时,我们首先要做的就是判断这两个对象是否相等。如果相等,那就好办了,因为这是属于重复元素,就不需要再进行存储了,但如果不相等,那就将这个新对象想别的办法存起来,那两个哈希冲突的对象,该怎么判断相等呢?那就得用到equals()方法了。

到这里也就能明白,为什么hashCode()和equals()要同时重写。因为hashCode()方法用来定位索引位置,以提高效率的同时可能会发生哈希冲突,当发生哈希冲突时,我们就得通过equals()方法来判断冲突的对象是否相等,如果只重写了hashCode()方法,那哈希冲突发生时,即使两个对象相等,也不会判定为重复,进而导致数组里会存储一大堆重复对象,但如果只重写了equals()方法,那两个相等的对象内存地址可不会相等,这样还是会造成重复元素的问题,所以两个方法最好一起重写。

;