String 對象可謂再熟悉不過了,與此相關的面試題經常會引出內存性能優化的問題,本篇主要以 new String("123") 創建了幾個對象為例記錄。
一、你能回答正確嗎
String a = "123";
如上定義的為常量;
String b = a +"456";
如上 b 為變量,為啥? 難道常量拼接常量得到的是變量嗎?
不是,常量拼接常量得到的依舊是常量。
但是此時將 a 作為引用,a 已經不再是常量了,是變量了,所以得到的 b 自然就是變量。
String b = "123" + "456";
此時 b 為常量。
如果給 a 加上修飾符 final ,那么 a 就是個常量,那么 b 就為常量了。
二、String定義常量和變量的區別
我們再來通過兩種寫法分析:
String a = "123";
String b = new String("123");
如上第1行,定義了一個常量 a ,第2行,通過關鍵字 new 的形式,創建了一個變量 b 。
我們結合之前學過的 JVm 再深入一些,第1行在常量池開辟了一塊空間,存放字符串 123,通過 a 對象指向這個常量對象。第2行由於使用了 new 關鍵字,所以會在堆空間中開辟一塊內存區域,在其中存放字符串 123,並把內存的地址賦予 b 變量。
所以, a==b 嗎?顯示是 false,一個是堆內存,一個是常量池。
如果將 a 修改成:
String a = new String("123");
那么,a==b 嗎?
依舊是 false。
為什么?只要通過 new 形式,自然是創建兩個對象,所以是 false,即便是他們的值是一致的。
三、String常量變量的總結
String 常量存放在常量池中,jvm處於優化考慮,會讓內容一致的對象共享內存塊,但變量是放在堆空間中的,new 定義的不同變量內存地址不同。
String 常量連接常量,還是常量,依舊用常量池管理,但常量連接變量就是變量了。
四、創建幾個對象的練習
以下幾種情況(均不考慮字符串在常量池中已存在的情況):
1、String a="123";
創建了1個對象
jvm在編譯階段會判斷常量池中是否有 "123" 這個常量對象如果有,a直接指向這個常量的引用,如果沒有會在常量池里創建這個常量對象。
2、String a=new String("123");
創建了2個對象
同情況1,jvm編譯階段判斷常量池中 "123"存在與否,進而來判斷是否創建常量對象,然后運行階段通過new關鍵字在java heap創建String對象。
3、String a="123"+"456";
創建了1個對象
jvm編譯階段過編譯器優化后會把字符串常量直接合並成"123456",所有創建對象時最多會在常量池中創建1個對象。
4、String a="123"+new String("456");
創建了4個對象
常量池對象"123" ,"456",new String("456")創建堆對象,還有一個堆對象"123456"。
最后練習參考文章:https://blog.csdn.net/baidu_27969827/article/details/79219708