java - String s=new String()與String s = ""的區別
問題描述
是不是定義字符串如果不使用new來初始化的話相同的字符串會被定義成一個引用
問題解答
回答1:JVM對于String的存儲有一點特殊的地方在于有一塊String常量池。這個常量池里面存著對String對象的引用。
比如,String s = 'abc'會先去String常量池中查找有沒有已經存在的引用,如果沒有,聲明的abc會直接生成一個String對象,并且會在String常量池中存入一個引用指向這個String對象。
之后直接聲明的字符串同樣也會遵循上面的步驟,所以第二次String s2 = 'abc'從String常量池中找到了一個引用指向第一次聲明的字符串對象。
而new String('abc')這樣會直接在堆中創建新的對象,不會進入String常量池。要把這樣的對象引用放入常量池中就涉及另一個String類的方法intern(),這個方法就是返回一個String對象的常量池引用。如果這個對象不在常量池中,就會把這個String對象放入常量池中并返回對應的對象引用。
題主第一個截圖的方法str2.intern() == str3.intern()這樣使用也會返回true,調用intern()返回的String常量池引用是同一個。
回答2:Java中所有非內置數據類型都是引用。
String s = new String('xx');這種方式會創建一塊內存空間,并使引用s指向它。
String s = 'xx';這種方式會使引用s指向一塊共享的空間。
使用new的方式創建時str2和str3指向的不同的內存空間,故str2和str3是不相等的。
直接使用字符串賦值時str2和str3指向是的相同的內存空間,故str2和str3是相等的。
可以使用str2.equals(str3)來比較字符串的內容。
回答3:一個是在常量池中存儲,一個是在堆中new出新對象
回答4:幫助你理解這個過程。java中有幾個點你可以記住下1.所有的字符串都會在常量池生成,對應的是CONSTANT_String_info,不可變。2.普通的對象幾乎都是在堆中生成(當然也有一些比較特殊的比如Class類的對象可能在方法區生成,這個看不同的虛擬機實現,虛擬機規范并沒有強制)3.==這個判定的時候,對于引用類型,說到底都是比對內存地址。
好了,有了上述的觀念。第一個問題中,String s1 = new String('aaa'); 當你new一個對象,jvm就會在堆上幫你開辟一個對象空間,而s1是存在你的本地變量表的,s1指向這個對象空間(可以暫時理解為s1存著對象空間的地址)。所以你new了兩個,是兩個不同的對象空間。==判斷當然不同啦,因為,s1和s2指向不同的空間。
第二個問題,上述第一點,每個字符串都只會在常量池中存在一份,所以str2指向這個常量池字符串的地址,str3也是指向這個常量池字符串的地址。==判斷自然是相同的。
相關文章:
1. 哭遼 求大佬解答 控制器的join方法怎么轉模型方法2. 在mybatis使用mysql的ON DUPLICATE KEY UPDATE語法實現存在即更新應該使用哪個標簽?3. Navicat for mysql 中以json格式儲存的數據存在大量反斜杠,如何去除?4. mysql - 表名稱前綴到底有啥用?5. 編輯成功不顯示彈窗6. mysql - 怎么生成這個sql表?7. sql語句 - 如何在mysql中批量添加用戶?8. 怎么php怎么通過數組顯示sql查詢結果呢,查詢結果有多條,如圖。9. mysql - 數據庫表中,兩個表互為外鍵參考如何解決10. mysql儲存json錯誤
