【HashMap】为什么用自定义的类做HashMap的Key时需要重写hashcode方法和equals方法

  • 【一】为什么有这个问题
  • 【二】Object类的中的hashcode方法和equals方法
  • 【三】重写hashcode
  • 【四】重写equals方法
  • 【五】hashmap中使用hashcode和equals方法


【一】为什么有这个问题

因为HashMap的key有个特征,就是key值不能重复,否则add的时候会覆盖。那么如果使用自定义的类做HashMap的key,就需要用到equals方法判断两个对象是否相等,判断的条件就是hashCode生成的哈希值。

Java中所有的类都是Object类的子类,所以自定义的类也是Object类的子类,Object中自带的hashcode方法和equals方法,如果不重写hashcode方法和equals方法,即使两个自定义类的对象内容完全相同,也不会判定为重复,这也就违背了HashMap的key不可重复的特点。

【二】Object类的中的hashcode方法和equals方法

Java中所有的类都是Object类的子类,Object中自带的hashcode方法,就是通过计算内存地址转换一个整数实现的,是一个native方法,返回值是int类型:

public native int hashCode();

Object中自带的equals方法,从源码中可以看出,就是将比较与被比较对象用==进行比较,用的还是比较hash值的方式

public boolean equals(Object obj) {
    return (this == obj);
}

【三】重写hashcode

如果不重写hashcode方法,那么即便是两个内容完全相同key对象,每次通过hashcode方法计算出的hash值也都是不一样的,因为是根据其对象所在内存地址计算的,因而做存储时也会被当作时不同的key;

重写hashcode方法该怎么写呢,这没有固定的格式,项目中看到很多中写法,列举两个

(1)如果是一个实体类,有唯一的字段id,那么可以利用该字段直接作为其hash值,如下

public class user{
    private int id;
    private String name;
    
    @override
    public int hashcode(){
        return this.id;
    }
}

(2)也可以是如下写法,因为String类中已经重写了hashcode方法,所以当name值相同时,其对应hash值一定相同

public class User{
    private int id;
    private String name;
    
    @override
    public int hashcode(){
        return id*name.hashcode;
    }
}

【四】重写equals方法

equals方法在上面已经介绍过了,其实就是用==进行比较,如果不重写那么即便是两个内容完全相同的对象,equals方法返回值依然是false,和hashcode类似也需要进行类似的方法重写:

(1)思路是先使用instanceof关键字判断这个被比较对象与对象是否属于一个类,返回值是true的话,再将二者转换成json字符串,最后通过String已经重写好了的equals方法进行比较,如下:

public class User{
    private int id;
    private String name;
    
    @override
    public boolean equals(Object object){
        if(object instanceof User){
            User compareObj = (User)object;
            String soureStr = JsonUtils.objectToJson(this);
            String targetStr = JsonUtils.objectToJson(compareObj);
            return soureStr.equals(targetStr);
        }
        return false;
    }
}

【五】hashmap中使用hashcode和equals方法

举例,将一个对象user对象作为key存入hashmap当中,会经历以下流程,能够看出在判断两个对象是否相等时,要经过以下步骤:

(1)是先去判断二者的hash值是否相同,因为计算和比较hash值所耗费代价要远低于通过equals进行比较,所以先比较hash值也可以提高效率;
(2)如果比较结果不相等,则直接返回false;
(3)如果比较结果相等,则使用equals方法进行比较;
(4)如果相等,则返回true,如果不相等,则返回fasle。

sping redis 哈希map 获取所有key 的值_java

总结:在hashmap中,对于两个对象的比较,都充分用到了key对象的hashcode方法和equals方法;

如果不重写hashcode方法,那么即便是两个内容完全一致的key对象,它仍会视为是不同的key,导致数据存取无法进行;

如果不重写equals方法,那么equals方法只是简单的将两个对象进行==比较,实际上也是通过对象间的内存地址进行比较,也导致上述情况的发生。