demo:
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 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。
