基于Java字符串 "==" 與 "equals" 的深入理解
雖然關(guān)于Java字符串“==”和“equals”的問(wèn)題是Java學(xué)習(xí)中的最基礎(chǔ)的一個(gè)問(wèn)題,但是仍然有不少剛剛學(xué)習(xí)Java的同學(xué)搞不清楚里面的原理,最近和朋友討論到這個(gè)問(wèn)題,所以寫(xiě)篇文章給大家分享一下我自己的理解。
首先讓我們看一個(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中基類(lèi)Obejct的equals方法跟“==”方法其實(shí)是一致的呢?只不過(guò)是String類(lèi)繼承Object類(lèi)(Java中所有的類(lèi)都是繼承Object類(lèi)的)后,重載了equal的方法,使得其成為了比較字符串中的內(nèi)容。
很好的理解了equals之后,現(xiàn)在讓我們來(lái)研究一下“==”方法的撲朔迷離。
“==”是Java中的運(yùn)算符,它比較的內(nèi)容是兩個(gè)對(duì)象的指針,也就是實(shí)際對(duì)象的地址。因此很容易理解e==a是返回true的。
我們?cè)賮?lái)看下c和d之間的比較,看到new關(guān)鍵詞,說(shuō)明c和d都是重新申請(qǐng)了一塊內(nèi)存地址,然后里面賦值為“test”,所以c==d返回的是false。同理都可以理解
a,c,d,f之間的“==”操作都返回的是false。
然后再來(lái)看看g= a + “”,雖然沒(méi)有看到new關(guān)鍵詞,但是由于String類(lèi)“+”操作符被重載之后,重載的方法之中一定會(huì)重新new一個(gè)String,所以也就變成了上面的情況。那么最難理解的就是a和b的比較了,其實(shí)這是Java編譯器的優(yōu)化所帶來(lái)的結(jié)果,java編譯器有一個(gè)叫字符串常量池的存儲(chǔ)區(qū)域,由于String a = "test"這條語(yǔ)句編譯之后,“test”就會(huì)存儲(chǔ)在這個(gè)字符串常量池中,那么再定義b的時(shí)候,b仍然指向這塊區(qū)域,所以說(shuō)a和b指向的仍然是同一塊區(qū)域。所以a==b返回的true。
相關(guān)文章
基于Java的Socket多客戶端Client-Server聊天程序的實(shí)現(xiàn)
這篇文章主要介紹了基于Java的Socket多客戶端Client-Server聊天程序的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-03-03Java多線程Thread基礎(chǔ)學(xué)習(xí)
每一個(gè)正在執(zhí)行的程序都是一個(gè)進(jìn)程,資源只有一塊,所以在同一時(shí)間段會(huì)有多個(gè)程序同時(shí)執(zhí)行,但是在一個(gè)時(shí)間點(diǎn)上,只能由一個(gè)程序執(zhí)行,多線程是在一個(gè)進(jìn)程的基礎(chǔ)之上的進(jìn)一步劃分,需要的朋友可以參考下2023-04-04SystemServer進(jìn)程啟動(dòng)過(guò)程解析
這篇文章主要為大家介紹了SystemServer進(jìn)程啟動(dòng)過(guò)程解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-07-07分布式系統(tǒng)下調(diào)用鏈追蹤技術(shù)面試題
這篇文章主要為大家介紹了分布式系統(tǒng)下調(diào)用鏈追蹤技術(shù)面試問(wèn)題合集,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步早日升職加薪2022-03-03java數(shù)據(jù)結(jié)構(gòu)實(shí)現(xiàn)機(jī)器人行走
這篇文章主要為大家詳細(xì)介紹了java數(shù)據(jù)結(jié)構(gòu)實(shí)現(xiàn)機(jī)器人行走,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-01-01Java復(fù)制一個(gè)對(duì)象并且不想復(fù)制其中的空值屬性問(wèn)題
這篇文章主要介紹了Java復(fù)制一個(gè)對(duì)象并且不想復(fù)制其中的空值屬性問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-08-08Springboot2.X集成redis集群(Lettuce)連接的方法
這篇文章主要介紹了Springboot2.X集成redis集群(Lettuce)連接的方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-07-07