雖然關(guān)于Java字符串“==”和“equals”的問題是Java學(xué)習(xí)中的最基礎(chǔ)的一個(gè)問題,但是仍然有不少剛剛學(xué)習(xí)Java的同學(xué)搞不清楚里面的原理,最近和朋友討論到這個(gè)問題,所以寫篇文章給大家分享一下我自己的理解。
首先讓我們看一個(gè)例子:
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”方法的機(jī)制和原理?!癳quals”想必大家都知道,就是比較字符串的內(nèi)容,由于以上程序中的所有字符串內(nèi)容都是“test”,所以用equals比較它們都會(huì)相等。但是你是否知道Java中基類Obejct的equals方法跟“==”方法其實(shí)是一致的呢?只不過是String類繼承Object類(Java中所有的類都是繼承Object類的)后,重載了equal的方法,使得其成為了比較字符串中的內(nèi)容。
很好的理解了equals之后,現(xiàn)在讓我們來研究一下“==”方法的撲朔迷離。
“==”是Java中的運(yùn)算符,它比較的內(nèi)容是兩個(gè)對(duì)象的指針,也就是實(shí)際對(duì)象的地址。因此很容易理解e==a是返回true的。
我們?cè)賮砜聪耤和d之間的比較,看到new關(guān)鍵詞,說明c和d都是重新申請(qǐng)了一塊內(nèi)存地址,然后里面賦值為“test”,所以c==d返回的是false。同理都可以理解
a,c,d,f之間的“==”操作都返回的是false。
然后再來看看g= a + “”,雖然沒有看到new關(guān)鍵詞,但是由于String類“+”操作符被重載之后,重載的方法之中一定會(huì)重新new一個(gè)String,所以也就變成了上面的情況。那么最難理解的就是a和b的比較了,其實(shí)這是Java編譯器的優(yōu)化所帶來的結(jié)果,java編譯器有一個(gè)叫字符串常量池的存儲(chǔ)區(qū)域,由于String a = "test"這條語句編譯之后,“test”就會(huì)存儲(chǔ)在這個(gè)字符串常量池中,那么再定義b的時(shí)候,b仍然指向這塊區(qū)域,所以說a和b指向的仍然是同一塊區(qū)域。所以a==b返回的true。