前言最近在突然想到了String字符串拼接問題,于是做了一個(gè)demo測試了一下,到底String類型的字符串在拼接的時(shí)候,哪種情況下會(huì)走會(huì)走StringBulider進(jìn)行字符串拼接,而哪種情況編譯器會(huì)對(duì)代碼進(jìn)行優(yōu)化?話不多說,先看demo
一.問題案例1
可以發(fā)現(xiàn),str == str2的結(jié)果為false,那么我們?cè)诳纯聪乱粋€(gè)例子。 案例2
這時(shí)候,兩個(gè)字符串對(duì)比的結(jié)果為true。 二.探究問題這時(shí)候,疑問就來了,為什么結(jié)果會(huì)不一致呢?利用在cmd窗口輸入javap -c TestDemo.class命令,對(duì)字節(jié)碼文件進(jìn)行反編譯,發(fā)現(xiàn)了問題所在? 
可以看到在案例1中,java代碼底層走了StringBuilder,進(jìn)行字符串拼接,然后調(diào)用了StringBuilder的toString方法。 
而案例2中,對(duì)class文件進(jìn)行反編譯,發(fā)現(xiàn)代碼出現(xiàn)了一點(diǎn)變化,并沒有走StringBuilder進(jìn)行字符串拼接。 三.總結(jié)案例1中,通過變量和字符串拼接,java是需要先到內(nèi)存找變量對(duì)應(yīng)的值,才能進(jìn)行完成字符串拼接的工作,這種方式j(luò)ava編譯器沒法優(yōu)化,只能走StringBuilder進(jìn)行拼接字符串,然后調(diào)用toString方法,當(dāng)然返回的結(jié)果和常量池中的111這個(gè)字符串的內(nèi)存地址是不一樣的,因此結(jié)果為false。 案例2中,直接在表達(dá)式里寫值,java不用根據(jù)變量去內(nèi)存里找對(duì)應(yīng)的值,可以在編譯的時(shí)候直接對(duì)這個(gè)表達(dá)式進(jìn)行優(yōu)化,優(yōu)化后的表達(dá)式從 "111" + "" 直接變成了 "111" ,兩個(gè)String類型的變量都指向了常量池的111字符串,因此結(jié)果為true;
作者:叫我不矜持
|