java - String s=new String()與String s = ""的區(qū)別
問題描述
是不是定義字符串如果不使用new來初始化的話相同的字符串會被定義成一個引用
問題解答
回答1:JVM對于String的存儲有一點(diǎn)特殊的地方在于有一塊String常量池。這個常量池里面存著對String對象的引用。
比如,String s = 'abc'會先去String常量池中查找有沒有已經(jīng)存在的引用,如果沒有,聲明的abc會直接生成一個String對象,并且會在String常量池中存入一個引用指向這個String對象。
之后直接聲明的字符串同樣也會遵循上面的步驟,所以第二次String s2 = 'abc'從String常量池中找到了一個引用指向第一次聲明的字符串對象。
而new String('abc')這樣會直接在堆中創(chuàng)建新的對象,不會進(jìn)入String常量池。要把這樣的對象引用放入常量池中就涉及另一個String類的方法intern(),這個方法就是返回一個String對象的常量池引用。如果這個對象不在常量池中,就會把這個String對象放入常量池中并返回對應(yīng)的對象引用。
題主第一個截圖的方法str2.intern() == str3.intern()這樣使用也會返回true,調(diào)用intern()返回的String常量池引用是同一個。
回答2:Java中所有非內(nèi)置數(shù)據(jù)類型都是引用。
String s = new String('xx');這種方式會創(chuàng)建一塊內(nèi)存空間,并使引用s指向它。
String s = 'xx';這種方式會使引用s指向一塊共享的空間。
使用new的方式創(chuàng)建時str2和str3指向的不同的內(nèi)存空間,故str2和str3是不相等的。
直接使用字符串賦值時str2和str3指向是的相同的內(nèi)存空間,故str2和str3是相等的。
可以使用str2.equals(str3)來比較字符串的內(nèi)容。
回答3:一個是在常量池中存儲,一個是在堆中new出新對象
回答4:幫助你理解這個過程。java中有幾個點(diǎn)你可以記住下1.所有的字符串都會在常量池生成,對應(yīng)的是CONSTANT_String_info,不可變。2.普通的對象幾乎都是在堆中生成(當(dāng)然也有一些比較特殊的比如Class類的對象可能在方法區(qū)生成,這個看不同的虛擬機(jī)實(shí)現(xiàn),虛擬機(jī)規(guī)范并沒有強(qiáng)制)3.==這個判定的時候,對于引用類型,說到底都是比對內(nèi)存地址。
好了,有了上述的觀念。第一個問題中,String s1 = new String('aaa'); 當(dāng)你new一個對象,jvm就會在堆上幫你開辟一個對象空間,而s1是存在你的本地變量表的,s1指向這個對象空間(可以暫時理解為s1存著對象空間的地址)。所以你new了兩個,是兩個不同的對象空間。==判斷當(dāng)然不同啦,因為,s1和s2指向不同的空間。
第二個問題,上述第一點(diǎn),每個字符串都只會在常量池中存在一份,所以str2指向這個常量池字符串的地址,str3也是指向這個常量池字符串的地址。==判斷自然是相同的。
相關(guān)文章:
1. 在mybatis使用mysql的ON DUPLICATE KEY UPDATE語法實(shí)現(xiàn)存在即更新應(yīng)該使用哪個標(biāo)簽?2. mysql - 數(shù)據(jù)庫建字段,默認(rèn)值空和empty string有什么區(qū)別 1103. mysql - 這種分級一對多,且分級不平衡的模型該怎么設(shè)計表?4. Navicat for mysql 中以json格式儲存的數(shù)據(jù)存在大量反斜杠,如何去除?5. mac OSX10.12.4 (16E195)下Mysql 5.7.18找不到配置文件my.cnf6. mysql mysql_real_escape_string() 轉(zhuǎn)義問題7. 新人求教MySQL關(guān)于判斷后拼接條件進(jìn)行查詢的sql語句8. mysql - 千萬數(shù)據(jù) 分頁,當(dāng)偏移量 原來越大時,怎么優(yōu)化速度9. MySQL FOREIGN KEY 約束報錯10. mysql - 數(shù)據(jù)庫表中,兩個表互為外鍵參考如何解決
