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} .