通常情况下,基本数据类型(byte,short,int,long,char,double,folat,boolean)比较时,我们使用“==”来进行比较;当两个对象比较时,我们通常使用超类Object的中的“equals”方法来比较两个对象!但实际情况Obejct的equals方法跟“==”方法其实是一致的呢?都是比较引用地址的:
public boolean equals(Object obj) {
return (this == obj);
}
只不过是String类继承Object类(Java中所有的类都是继承Object类的)后,重写了equal的方法,使得其成为了比较字符串中的内容而不是地址,这就是String的equals(比内容)和==(比地址)的区别;jdk当中的String、Math等封装类都对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;
}
对于这个代码段:if (v1[i++] != v2[j++])return false;我们可以非常清晰的看到String的equals()方法是进行内容比较,而不是引用比较。
下面我们一起来看看Stirng类中equals和==具体实例:
public class StringDemo {
public static void main(String[] args) {
//作为基本类型使用
String a = "123";
String b = "123";
System.out.println("------------a和b比较------------");
System.out.println("a==b:" + (a == b));// true
System.out.println("a.equals(b):" + a.equals(b));//true
//作为对象使用
String c = new String("123");
String d = new String("123");
String e = d;
String f =a+"";
// c和d比较
System.out.println("------------c和d比较------------");
System.out.println("c==d:" + (c == d));// false
System.out.println("c.equals(d):" + c.equals(d));// true
// e和d比较
System.out.println("------------e和d比较------------");
System.out.println("e == d:" + (e == d));// true
System.out.println("e.equals(d):" + e.equals(d));// true
// e和c比较
System.out.println("------------e和c比较------------");
System.out.println("e == c:" + (e == c));//false
System.out.println("e.equals(c):" + e.equals(c));// true
//a和c比较
System.out.println("------------a和c比较------------");
System.out.println("a == c:" + (a == c));// false
System.out.println("a.equals(c):" + a.equals(c));// true
//a和f比较
System.out.println("------------a和f比较------------");
System.out.println("a == f:" + (a == f));// false
System.out.println("a.equals(c):" + a.equals(f));// true
}
}
上面的实例,我们通常会疑惑a==b为什么会是true?下面来解释一下:
“==”是Java中的运算符,它比较的内容是两个对象的指针,也就是实际对象的地址。因此很容易理解e==d是返回true的。
我们再来看下c和d之间的比较,看到new关键词,说明c和d都是重新申请了一块内存地址,然后里面赋值为“test”,所以c==d返回的是false。
然后再来看看f= a + “”,虽然没有看到new关键词,但是由于String类“+”操作符被重载之后,重载的方法之中一定会重新new一个String,所以也就变成了上面的情况。
敲黑板:那么最难理解的就是a和b的比较了,其实这是Java编译器的优化所带来的结果,java编译器有一个叫字符串常量池的存储区域,由于String a = "123"这条语句编译之后,“test”就会存储在这个字符串常量池中,那么再定义b的时候,b仍然指向这块区域,所以说a和b指向的仍然是同一块区域(这里有种把String当成基本数据类型用了的感觉),这样是为了节省内存空间。所以a==b返回的true