以下代码的运行结果是怎样的呢?
public class equalsTest {
public static void main(String[] args) {
String s1 = "aaa";
String s2 = "aaa";
String s3 = new String("aaa");
System.out.println(s1==s2);
System.out.println(s1==s3);
System.out.println(s1.equals(s3));
System.out.println(s1.hashCode());
System.out.println(s2.hashCode());
System.out.println(s3.hashCode());
}
}
在解答之前,我们应该了解的:
1.在Java中,基本数据类型中存储的是数据值,引用数据类型存储的是对象的地址值。
2.Java内存中,String字符串在方法区开辟了一个常量池,每声明一个String类型的引用并初始化时,首先在常量池中寻找是否已有初始化的值,如果有,引用直接指向这个字符串的地址,如果没有则先在常量池中创建这个字符串,然后将引用指向它。
4.equals()方法是Object的方法,没重写之前比较的是对象的地址。重写之后可根据需要比较对象的值,String类重写了equals()方法。
5.对象创建时,在new之后,就会在堆内存空间开辟一块空间。
再次回看代码:
s1在常量池中创建了“aaa”字符串,并指向了"aaa"
s2在创建时查看常量池,发现已有"aaa"字符串,所以直接指向了"aaa"
s3使用了new关键字,在堆内存中创建了一个对象,对象中存放常量池中"aaa"的地址。
在理解之后,查看结果:
现在再来看问题,我们就不难知道:
1.==是比较运算符,在比较引用数据类型时,比较的是地址。
s1,s2指向同一个地址;s3指向对象地址中存放的内容虽然是常量池中"aaa"的地址,但他本身指向的是这个对象,在和s1比较时地址是不同的。
2.equals()作为一个方法,只能比较引用数据类型。重写之前比较的是地址的值,重写之后比较对象的属性。
再来理解equals方法,查看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;
}
equals()方法先使用==比较地址是否相同,如果相同,直接返回true;如果不同,再比较类型是否一致,如果一致再比较长度,若长度一致最后逐个比较每个字符。所以在对象的比较中,我们应该使用equals()比较值是否相同。
其实在对象的比较中,还有一个hashCode()方法,在查看结果时,我们发现三个变量使用hashCode()方法后,得到的结果是一致的,查看源码:
public int hashCode() {
int h = hash;
if (h == 0 && value.length > 0) {
char val[] = value;
for (int i = 0; i < value.length; i++) {
h = 31 * h + val[i];
}
hash = h;
}
return h;
}
hashCode()方法直接用对象的值做运算,若两个对象值一致,那么hashCode()方法得到的结果也应该一致。所以在比较中,我们也可以用hsahCode()方法比较对象的值。