问题:两个对象值相同(x.equals(y) == true),但是可能存在hashCode不同吗?
==的定义
比较的是两个对象的内存地址,相等则意味着内存地址一样。
对象的equals方法
Object#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 anotherString = (String)anObject;
int n = value.length;
if (n == anotherString.value.length) {
char v1[] = value;
char v2[] = anotherString.value;
int i = 0;
while (n-- != 0) {
if (v1[i] != v2[i])
return false;
i++;
}
return true;
}
}
return false;
}
JDK 中不止 String 类重写了equals 方法,还有数据类型 Integer,Long,Double,Float等基本也都重写了 equals
方法。所以我们在代码中用 Long 或者 Integer 做业务参数的时候,如果要比较它们是否相等,记得需要使用 equals
方法,而不要使用 ==
。
对象的HashCode
ava中的hashCode方法就是根据一定的规则将与对象相关的信息(比如对象的存储地址,对象的字段等)映射成一个数值,这个数值称作为散列值。
问题解答
- 使用
==
这个表达式判断,如果返回true
,意味着两个对象的hashCode一定相同。 - 理论情况下,
x.equals(y)==true
,如果没有重写equals
这个方法,这两个对象的内存地址是是相同的,也就意味着hashCode必然也相等。但可能出现重写equals的方法的情况
public class App
{
public static void main( String[] args ) {
A a = new A();
B b = new B();
System.out.println(a.equals(b));
System.out.println(a.hashCode() + "," + b.hashCode());
}
}
class A {
@Override
public boolean equals(Object obj) {
return true;
}
}
class B {
}
- hashCode相等,equals方法是不一定相等的。因为HashMap映射的HashCode可能相同,但是euqals判断不一样,会在链表中添加元素。
- 如果重写了equals方法,没有重写HashCode,当前对象作为主键插入HashMap中,以另一个判断equals为true的对象作为主键获取对象,是获取不到的。
- 在实际开发中,必须遵循重写
equals
方法的同时也需要重写hashCode
方法这一原则