为什么要重写 equals和hashcode 方法

       equals()和hashcode()都继承于Object,并且Object都提供了默认实现,我们先来看一下object类中的这两个方法.

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

        此处可知,默认情况下equals()方法和 "==" 号的作用是相同的,在比较基础数据类型时比较的是值,在比较对象时,则是比较对象的地址值. 我们再来看一下 hashcode()方法

    @HotSpotIntrinsicCandidate
    public native int hashCode();


       哈希值的具体算法规则,我们不再这里引申,简而言之,在不重写hashcode()和equals()的情况下,我们所创建的不同对象他们用equals()方法比较时,在不考虑哈希冲突的情况下,他们的哈希值也必不可能相同.因此在将对象数据放到set集合中时,是不会达到去重的效果的. 那么就会出现与我们预期相违背的情况.

        这边简单的做一个代码演示,我们先随便创建一个实体类Person

public class Person {
    private String name;
    private int age;

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

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}
public class Test {
    public static void main(String[] args) {
        Person p1 = new Person("张三",18);
        Person p2 = new Person("张三",18);

        //结果为false
        System.out.println(p1.equals(p2));
        

        HashSet<Person> set = new HashSet<>();
        set.add(p1);
        set.add(p2);
        //遍历结果能得到两个相同的Person{name='张三', age=18}
        set.forEach(System.out::println);
    }
}

         我们创建两个姓名和年龄都相同的Person对象,在正常的逻辑下,我们可以清楚的知道这两个对象应该是相同的,但是因为没有重写equals()和hashcode()方法,所以我们比较了两个对象的地址值,自然是得到了false的结果.同样如果将这两个对象放进一个Hashset集合或者Hashmap集合中是达不到去重效果的,在遍历set集合时,我们会得到两个相同的Person{name='张三', age=18}.

        这也就是为什么要重写hashCode和equals方法的原因了,因为我们更期望的是,当来两个不同的对象,但是他们的属性值是完全一样的,我们认为这是同一个东西. 在放进一个set集合时也可以达到去重的效果. 我们重写equals()和hashcode()方法,代码再演示一下

package com.itheima.boot;

import java.util.Objects;

public class Person {
    private String name;
    private int age;

    public Person(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;
        Person person = (Person) o;
        return age == person.age && Objects.equals(name, person.name);
    }

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

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}

        测试代码不再重复演示,我们会发现,经过重写后,System.out.println(p1.equals(p2)); 的结果会变成true,遍历set集合在控制台也只会输出一个 Person{name='张三', age=18} .

    

        

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值