为什么重写equals方法必须要重写hashCode方法

为什么重写equals方法必须要重写hashCode方法

​ 了解这个问题之前我们得需要知道hashCode的作用。equals方法和hashCode方法都是Object类中的基础方法,用来判断两个对象是否相等。而平时我们比较两个对象不是用equals来比较吗?为什么又需要hashCode呢?原因就在于hashCode能够提升性能。
​ 例如我们常用的HashMap这个容器来说,它是通过键hashCode来计算出索引位置(存储位置)的,如果没有hash那就只能遍历元素相比下效率低于hash定位。
​ 那为什么不能直接使用hashCode就确定元素相等呢?下面看一段代码
为什么重写equals方法必须要重写hashCode方法_第1张图片
​ 通过这段代码可以看出不同的对象hashCode可能一样,那为什么重写了equals方法要重写hashCode呢?原因就是比较对象相等首先要比较hashCode是否相等,如果hashCode相等才会去比较equals,下面通过代码来验证

/**
 * 定义一个实体类
 * @Data注解为lombok注解也可以自己写getter和setter方法
 */
@Data
class A {
    private String name;
    private int age;

    public A(String name, int age) {
        this.name = name;
        this.age = age;
    }

    @Override
    public boolean equals(Object o) {
        if (this == o) {
            return true;
        }
        if (o == null || getClass() != o.getClass()) {
            return false;
        }
        A a = (A) o;
        return age == a.age && Objects.equals(name, a.name);
    }
}

​ A类中只重写了equals方法并没有重写hashCode方法,下面使用A作为HashMap的键来测试

public class Test {
    public static void main(String[] args) throws Exception {
        Map<A, Integer> map = new HashMap<>();
        A a1 = new A("zhangsan", 12);
        A a2 = new A("zhangsan", 12);
        // 调用equals方法判断相等
        System.out.println(a1.equals(a2));
        map.put(a1, 1);
        map.put(a2, 2);
        System.out.println("-------------");
        map.forEach((k, v) -> System.out.println(k + "---> " + v));
    }
}

为什么重写equals方法必须要重写hashCode方法_第2张图片
​ 运行代码可以看出对象相等但是map中的键并没有覆盖,可见在比较对象相等前先会判断对象hashCode是否相等,下面我们把A类中的hashCode方法一起重写

/**
 * 定义一个实体类
 * @Data注解为lombok注解也可以自己写getter和setter方法
 */
@Data
class A {
    private String name;
    private int age;

    public A(String name, int age) {
        this.name = name;
        this.age = age;
    }

    @Override
    public boolean equals(Object o) {
        if (this == o) {
            return true;
        }
        if (o == null || getClass() != o.getClass()) {
            return false;
        }
        A a = (A) o;
        return age == a.age && Objects.equals(name, a.name);
    }

    @Override
    public int hashCode() {
        return Objects.hash(name, age);
    }
}

​ 再次运行代码结果如下:

为什么重写equals方法必须要重写hashCode方法_第3张图片

​ map中的键发生了覆盖,出现以上问题的原因是,如果只重写了 equals 方法,那么默认情况下,会先判断两个对象的 hashCode 是否相同,此时因为没有重写 hashCode 方法,所以会直接执行 Object 中的 hashCode 方法,而 Object 中的 hashCode 方法对比的是两个不同引用地址的对象,所以结果是 false,那么 equals 方法就不用执行了,直接返回的结果就是 false:两个对象不是相等的,于是就在 map集合中插入了两个相同的对象的键

小结

​ hashCode 和 equals 两个方法是用来协同判断两个对象是否相等的,采用这种方式的原因是可以提高程序插入和查询的速度,如果在重写 equals 时,不重写 hashCode,就会导致在某些场景下,例如将两个相等的自定义对象存储在 map集合时,就会出现程序执行的异常,为了保证程序的正常执行,所以我们就需要在重写 equals 时,也一并重写 hashCode 方法才行。

你可能感兴趣的:(java,java,hash)