日常整理:为什么我们重写equals的时候也要重写hashcode

如果我们使用了map等的容器,并且key采用的是自定义的对象,那这个是一定要重写的

先说必要性,重写equals是为了判断实例是否相等,而重写hashcode是为了快速判重

先看一下这几个比较重要的点

- 如果两个对象相等,那么它们的 hashcode 一定相同

- 如果两个对象相等,那么 equals() 方法返回的一定是true

- 如果两个对象 hashcode 相同,它们本身并不一定相等

所以,如果equals方法被重写了,hashcode也应该被重写

hashcode本身的动作是在堆上的对象产生了一个独特的值,而如果没有重写hashcode,那么两个class对象无论如何都不会相等,就算这两个class对象指向了相同的数据

我们来做个测试,只重写equals

    public static void main(String[] args) throws Exception {
        Info info = new Info("test");
        Info info1 = new Info("test");
        Map<Info, String> map = new HashMap<>(2);
        map.put(info, "对象1");
        map.put(info1, "对象2");
        System.out.println("map = " + map);
    }


    static class Info{
        private String message;
        public Info(String message) {
            this.message = message;
        }

        @Override
        public String toString() {
            return "Info{" +
                    "message='" + message + '\'' +
                    '}';
        }

        @Override
        public boolean equals(Object o) {
            if (this == o) return true;
            if (o == null || getClass() != o.getClass()) return false;
            Info info = (Info) o;
            return Objects.equals(message, info.message);
        }

//        @Override
//        public int hashCode() {
//            return Objects.hash(message);
//        }
        
    }

可以看到结果

我们再试试把hashcode的注释放开

我们重写的equals的逻辑是,只要message相等,那么对象就相等,but,从map的运行结果看,都放进去了,so,我们来看看hashmap

通过源码我们知道, hash不一样的话,可以直接插入到数组中,但是,因为我们没重写hashcode,所以调用了object的hashcode,而object的hashcode,是通过对象在堆里的地址计算出一个int值,所以,创建的不同对象的hashcode肯定是不同的,所以才会两个都插入到了map里

-------------

我们反一下,如果重写了hashcode,是不是要重写equals

这个当然是的

如果两个对象的hashcode相同,map在put的时候,hash相同,找到的位置就相同,取到了数据,进行equals比较,如果没有重写equals,那么就是对象地址在比较,结果就是false了,不信试试

 map在get的时候,是根据hashcode找,找到了,不止一个,那它就会用equals去找相等的值

所以说,我们在使用中,如果重写了equals,也要重写hashcode

如果你仔细观察,你会发现IDE其实也是放在一起的,比如我用IDEA

 总之,用的时候一定要小心哦

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值