雖然關於Java字串「==」和「equals」的問題是Java學習中的最基礎的一個問題,但是仍然有不少剛學習Java的同學搞不清楚裡面的原理,最近和朋友討論到這個問題,所以寫篇文章跟大家分享一下我自己的理解。
首先讓我們來看一個例子:
複製代碼代碼如下:
public class TestStringEquals {
public static void main(String[] args) {
String a = "test";
String b = "test";
String c = new String("test");
String d = new String("test");
String e = a;
String f = new String(a);
String g = a + "";
System.out.println(a == b ? "expression /"a==b/" is true"
: "expression /"a==b/" is false");
System.out.println(a == c ? "expression /"a==c/" is true"
: "expression /"a==c/" is false");
System.out.println(c == d ? "expression /"c==d/" is true"
: "expression /"c==d/" is false");
System.out.println(a == e ? "expression /"a==e/" is true"
: "expression /"a==e/" is false");
System.out.println(a == f ? "expression /"a==f/" is true"
: "expression /"a==f/" is false");
System.out.println(a == g ? "expression /"a==g/" is true"
: "expression /"a==g/" is false");
if (a.equals(b) && b.equals(c) && c.equals(d) && d.equals(e)
&& e.equals(f) && f.equals(g)) {
System.out
.println("a equals b equals c equals d equals e equals f equals g");
}
}
}
如果先不答案,大家試著去做一做,能否保證全部正確呢?
以下公佈一下答案:
複製代碼代碼如下:
expression "a==b" is true
expression "a==c" is false
expression "c==d" is false
expression "a==e" is true
expression "a==f" is false
expression "a==g" is false
<div style="text-align: left;"></div>a equals b equals c equals d equals e equals f equals g
要很好的理解這種題目,最好的方法就是深入理解Java中String的「==」方法和「equals」方法的機制和原理。 “equals”想必大家都知道,就是比較字串的內容,由於以上程式中的所有字串內容都是“test”,所以用equals比較它們都會相等。
但你是否知道Java中基類Obejct的equals方法跟「==」方法其實是一致的呢?只不過是String類別繼承Object類別(Java中所有的類別都是繼承Object類別的)後,重載了equal的方法,使得其成為了比較字串中的內容。
在很好的理解了equals之後,現在讓我們來研究一下「==」方法的撲朔迷離。
「==」是Java中的運算符,它比較的內容是兩個物件的指針,也就是實際物件的位址。因此很容易理解e==a是回傳true的。
我們再來看下c和d的比較,看到new關鍵字,說明c和d都是重新申請了一塊記憶體位址,然後裡面賦值為“test”,所以c==d回傳的是false。同理都可以理解
a,c,d,f之間的「==」操作都回傳的是false。
然後再來看看g= a + “”,雖然沒有看到new關鍵字,但是由於String類別“+”運算符被重載之後,重載的方法之中一定會重新new一個String,所以也就變成了上面的情況。那麼最難理解的就是a和b的比較了,其實這是Java編譯器的最佳化所帶來的結果,java編譯器有一個叫字串常數池的儲存區域,由於String a = "test"這條語句編譯之後,「test」就會儲存在這個字串常數池中,那麼再定義b的時候,b還是指向這塊區域,所以說a和b指向的仍然是同一塊區域。所以a==b回傳的true。