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. docker - 各位電腦上有多少個容器啊?容器一多,自己都搞混了,咋辦呢?2. MySQL數據庫中文亂碼的原因3. macos - mac下docker如何設置代理4. docker不顯示端口映射呢?5. 關docker hub上有些鏡像的tag被標記““This image has vulnerabilities””6. android studio總是在processes running好久7. docker-compose 為何找不到配置文件?8. css - C#與java開發Windows程序哪個好?9. mysql - 新浪微博中的關注功能是如何設計表結構的?10. docker gitlab 如何git clone?
