为什么重写equals时必须重写hashCode方法?

​ 以java 1.8 的 HashMap为例,不管我们是在put,get还是remove时,都要先经过hash方法定位key存在的Node数组位置,然后遍历该位置的红黑树或链表,使用equals方法进行比较,而hashcode()方法默认是对堆上的对象产生独特值.

//Object.class
/**
 * Returns a hash code value for the object. 
 */
public native int hashCode();

​ 如果没有重写hashCode方法,则两个对象无论如何也不会相等,即使两个对象实际是相等的。所以这样就会导致hashmap失效,因为第一步的定位都不同,更别提第二步用equals进行判断了。那重写了hashCode方法后是什么样的呢,这里我们拿链表节点ListNode举例

import java.util.Objects;
public class ListNode {
    int val;
    ListNode next = null;
    ListNode(int val) {
        this.val = val;
    }
    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        ListNode listNode = (ListNode) o;
        return val == listNode.val &&
                Objects.equals(next, listNode.next);
    }
    @Override
    public int hashCode() {
        return Objects.hash(val, next);
    }
}

​ 重写的hashCode实际上调用的Objects.hash(),并传入了当前类的属性值作为参数,查看后我们发现,实际上重写了hashcode方法就是对当前类的每个属性计算对应的hashcode值, 并且通过计算公式①得到最后的hashcode值。这样就能保证两个对象如果属性值都相同则他们的hashcode值也相同。

//Objects.hash()
public static int hash(Object... values) {
    return Arrays.hashCode(values);
}

//Arrays.hashCode()
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;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值