前言
学到集合这一章,老师讲到Set集合中的元素是不重复的,手段则是通过重写hashCode()和equals()方法。本贴通过源码分析下是如何实现的。
先看下Set集合存入数据的大致流程:
先通过hashCode()方法得到元素的哈希值,通过哈希值计算对象的存储位置
这个流程中用到了hashCode和equals方法。
在Object中的hashCode方法是通过native关键字调用底层其他语言的方法进行实现的。
public native int hashCode();
这个方法是通过地址计算的对象的哈希值,也就是不同地址的对象哈希值不同。但是对于使用new关键字创建的对象,他们的地址肯定是不同的,但是其内容可能相同的。对于系统来说,我们关心的是对象的内容而不是地址,所以需要重写hashCode方法。
//重写hashCode方法前
Student s1 =new Student("张三");
Student s2 =new Student("张三");
System.out.println(s1);//Day06.Student@3f3afe78
System.out.println(s2);//Day06.Student@7f63425a
System.out.println(s1.hashCode());//1060830840
System.out.println(s2.hashCode());//2137211482
//重写hashCode方法后
Student s1 =new Student("张三");
Student s2 =new Student("张三");
System.out.println(s1);//Day06.Student@bd2e9
System.out.println(s2);//Day06.Student@bd2e9
System.out.println(s1.hashCode());//774889
System.out.println(s2.hashCode());//774889
我们看到,重写方法后,相同对象的哈希值是相同的。但是,只比较哈希值不靠谱的,因为有的时候内容不同,哈希值也可能相同。
System.out.println("重地".hashCode());//哈希值:1179395
System.out.println("通话".hashCode());//哈希值:1179395
对于最底层的hashCode来说,哈希值不同,对象肯定不同,但是对象的哈希值相同时,还需要比较其内容。这里也就是用到了equals方法
Object类是所有类的根,它里边的equals()方法是比较地址是否相同的,也就是说原始equals方法是用来比较地址的:
public boolean equals(Object obj) {
return (this == obj);
}
那为什么我们比较字符串的时候是比较内容呢?因为String类中重写了equals()方法,当然此时我是不能完全看明白,只能大概理解这是比较内容是否相同
public boolean equals(Object anObject) {
if (this == anObject) {
return true;
}
if (anObject instanceof String) {
String aString = (String)anObject;
if (coder() == aString.coder()) {
return isLatin1() ? StringLatin1.equals(value, aString.value)
: StringUTF16.equals(value, aString.value);
}
}
return false;
}
结论
Java提供的类是重写了hashCode()和equals()方法是被重写的,我们可以只用拿来用,但是自己定义的类比如Student类,Animal类一定要重写这两个方法。IDEA中是提供了自动生成的方法的,如下
String name;
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
Student student = (Student) o;
return name != null ? name.equals(student.name) : student.name == null;
}
@Override
public int hashCode() {
return name != null ? name.hashCode() : 0;
}