请尊重作者劳动成果,转载请注明出处:http://my.oschina.net/javaSmiler
如有疑问及讨论欢迎大家联系本作者email_cpf@163.com
String 为什么要重写equals()
1.先看此代码
String s1 = "hello world";
String s2 = new String("hello world");
System.out.println("s1==s2:"+s1.equals(s2));
Object o1 = new Object();
Object o2 = new Object();
System.out.println("o1==o2:"+o1.equals(o2));
控制台输出为:
s1==s2:true
o1==o2:false
那么问题就来了,本来就是两个New 出来的对象,按照常理不是在栈(stack)是对象的声明,堆(stack)中是对象的实例吗?中怎么可能String 和 Object 的方法会有这么大差异昵?
查看源码:
Object equlas() 方法:
public boolean equals(Object obj) {
return (this == obj);
}
说明返回的是对象中的引用比较,所以o1和02的对象引用当然不相等
String equlas()方法
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;
}
从源码可看出三点:1.首先比较字符串s1和s2的引用是否相等,相等则返回(引用相同,则堆中的对象实例必然也相同);
2.如果引用不同,那么比较String中的内容是否相同,如果相同则返回true.
3.String 的本质就是字符数据,看源码即可得知。
不难理解s1==s2为什么能相等了。
那么String为什么要重新equlas方法呢?
这里涉及到一些jvm的知识,请参看http://blog.csdn.net/zhangerqing/article/details/8214365中的内存结构部分。
大家知道String 是 java中的一个重要变量,如果每一个程序运行 ,jvm都给String new出来一个堆内存,无疑是浪费。
如果避免这种浪费,就用到了jvm方法区中的常量池。专门来存储字符串的内容。每一个new 一个对象或者 申明一个时,
JVM都会去方法区的常量池中去找是否包含“hello world”,这个字符串,如果有那么返回相同的引用地址。
内存图如下:
反向思维:如果String不重写equals()方法,而是调用父类Object的equals方法。那么我们只是知道两个引用不同,里面内容无法得知是否相等,
在常用的开发中,我们对于String更加关系的是两个字符串中的内容是否相同,而不是是不是同一个引用。所以结合常量池,String有理由重写equals方法。