核心两句话:
- ==在比较对象的时候,是比较他们是不是引用的同一个对象,即比较对象的地址
- 而equals比较对象的“值”是否相等,取决于equals方法的实现(一定要看如何实现的才能判断equals的返回结果)
经典问题:String类
情况1:
String s1="abc";
String s2="abc";
System.out.println(s1==s2); // true
System.out.println(s1.equals(s2)); // true
分析:
第三行为true的原因: JVM的知识告诉我们,"abc"被分配到常量池,然后是s1变量值是"abc"在常量池中的地址,
然后再创建s2的时候,发现"abc"已经存在,所以直接将该地址赋值给s2,所以s1==s2结果为true,如下图:
第四行为true分析:
首先看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;
}
发现源码是一个字符一个字符的比较String类对象的值是否相等,结合上图可见 "abc"在常量池,s1和s2当然相同,我们作如下改动:
String s3=new String("abc");
System.out.println(s1==s3); //false
System.out.println(s1.equals(s3)); //true
显然,符合我们的分析,s3是new出来的对象,在java堆上,所以s3和s1 的地址肯定不相同,但是值相同
至此,似乎一切都很完美,但是真的是这样吗,看下面的例子:
Integer i1=2;
Integer i2=2;
Integer i3=new Integer(2);
System.out.println(i1.equals(i2)); // true
System.out.println(i1.equals(i3)); // true
System.out.println(i1==i2); // true
System.out.println(i1==i3); // false
结果很显然,不符合我们上面的总结,i1和i2是两个不同的局部变量,应该在java栈上有不同的地址,但是为什么 i1==i2 的结果是true???
然后,我们改变一下值:
Integer i1=200;
Integer i2=200;
Integer i3=new Integer(200);
System.out.println(i1.equals(i2)); // true
System.out.println(i1.equals(i3)); // true
System.out.println(i1==i2); // false
System.out.println(i1==i3); // false
第9行代码结果变成了false,为什么???
是不是很神奇,这是因为Integer类有一个缓冲池一样的东西,当Integer类对象型的值范围是 -128~127时,就从缓冲池取出这个值,即将这个值在缓冲池中的地址赋值给这个变量,当nteger类对象型的值超过这个范围的时候才会重新开辟出一块内存没把这个值写进去,然后把这块内存的地址赋值给变量,所以当变量改为200时,i1==i2 值为false。
综上所述,你应该已经明白==和equals的区别,所以你即便你不太清楚他们的区别,也应该知道该用什么。
不知道?答案当然是用equals,因为绝大多数时候我们只是需要比较两个变量的值是否相等没然后做相应的处理,极少时候会用到比较两个对象的地址,除非是做特别底层的开发和优化。
结束语:源码之下,了无秘密