基础 | 重写hashCode方法

参加美团秋招面试时,被问到 「如何重写hashCode方法?重写hashCode()方法需要注意什么?」,在此做一个系统的总结与梳理,同时也填一下之前埋下的坑,哈哈。


参考答案

为什么要重写hashCode()方法?

Object类中hashCode()方法默认是将对象的存储地址进行映射,并返回一个整形值作为哈希码。

若重写equals()方法,使其比较两个对象的内容,并保留hashCode()方法的默认实现,那么两个明明「相等」的对象,哈希值却可能不同。

如果两个对象通过equals()方法比较的结果为true,那么要保证这两个对象的哈希值相等。

因此,在重写equals()方法时,建议一定要重写hashCode()方法。

如何重写hashCode()方法?

由于Object类的 hashCode() 方法是本地的(native),故其具体实现并不是由Java所完成的。

需要实现hashCode()方法时,可以直接调用Objects.hash(Object… values)方法来获取对应的哈希值。其内部的具体实现是调用Arrays.hashCode(Object[])方法来完成的。

Arrays中计算哈希值的核心代码如下:

public static int hashCode(Object a[]) {
    if (a == null)
        return 0;

    int result = 1;

    for (Object element : a)
        result = 31 * result + (element == null ? 0 : element.hashCode());

    return result;
}

那么,为什么选择数字31作为计算哈希值的因数

The value 31 was chosen because it is an odd prime. If it were even and the multiplication overflowed, information would be lost, as multiplication by 2 is equivalent to shifting. The advantage of using a prime is less clear, but it is traditional. A nice property of 31 is that the multiplication can be replaced by a shift and a subtraction for better performance: 31 * i == (i << 5) - i. Modern VMs do this sort of optimization automatically. ------引用自《Effective Java》

重写hashCode()方法需要注意什么?

  • 应用程序执行期间,只要一个对象用于equals()方法的属性未被修改,则该对象多次返回的哈希值应相等。
  • 如果两个对象通过equals()方法比较的结果为true,那么要保证这两个对象的哈希值相等。
  • 如果两个对象通过equals()方法比较的结果为false,那么这两个对象的哈希值可以相等也可以不相等,但理想情况下是应该不相等,以提高散列表的性能。

源码阅读

// Object类中hashCode()方法的默认实现
public native int hashCode();
// Arrays中计算哈希值的核心代码
public static int hashCode(Object a[]) {
    if (a == null)
        return 0;

    int result = 1;

    for (Object element : a)
        result = 31 * result + (element == null ? 0 : element.hashCode());

    return result;
}

推荐阅读


欢迎关注

Java名企面试吧,每天10点24分,我们不见不散!

丙子先生的宗旨是,每天以短篇幅讲高频面试题,不增加太多负担,但需要持之以恒。

能力有限,欢迎指教!

Java中,`hashCode()` 方法是一个Object类的方法,用于生成一个对象的哈希码(HashCode),这个哈希码通常用来进行散列查找(如HashMap、HashSet等数据结构)。当你创建自定义类并覆盖 `hashCode()` 和 `equals()` 方法时,需要注意保持这两个方法的一致性,即对于两个相等的对象,它们的哈希码必须相同;而对于不相等的对象,它们的哈希码可能不同,但相差不能过大。 如果你正在使用Java 17并且想要重写 `hashCode()` 方法,以下是一些基本步骤: 1. **理解基础原则**: - 哈希码应根据对象的内容计算,而不是对象引用本身。 - 尽量保证相同的对象产生相同的哈希码,不同的对象产生不同的哈希码。 - 如果类中有多个字段,你可以选择按某个或几个字段的组合来计算哈希码,例如使用 `Objects.hash()` 方法可以帮助简化过程。 ```java int hashCode = Objects.hash(field1, field2); ``` 2. **处理null值**: - null 不应该被包含在哈希码计算中,因为null通常用作“未设置”或“不存在”的标志。 3. **不可变性**: - 对于不可变对象,`hashCode()` 应该始终返回相同的值,即使对象没有改变。这是因为不可变对象一旦创建,其状态就不会变化。 4. **一致性**: - 当修改了对象的状态,确保调用 `equals()` 后重新计算 `hashCode()`,以便维护正确的行为。 5. **考虑哈希冲突**: - 避免过度密集的哈希碰撞(大量对象拥有相同的哈希码),这可能导致性能下降。可以通过增加哈希桶的数量或者采用更复杂的哈希算法来改善。 6. **实现**: ```java @Override public int hashCode() { // 检查对象是否为null if (this == null) { return 0; // 或者使用Objects.hash(null)的结果 } // 使用Objects.hash或其他方法计算基于字段的哈希码 int result = Objects.hash(field1, field2, ...); return result; } ``` 记得在编写 `equals()` 方法时也一起考虑,确保当两个对象的 `hashCode()` 相同时,它们确实相等。这是实现equals-hashCode一致性的关键。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值