<code>String a=</code><code>"a"</code><code>+</code><code>"b"</code><code>+</code><code>"c" 建立了幾個對象</code>
<code></code>
這個問題涉及到了字元串常量池和字元串拼接
通過編譯器優化後,得到的效果是
Java中字元串對象建立有兩種形式,一種為字面量形式,如<code>String str = "droid";</code>
另一種就是使用new這種标準的構造對象的方法,如<code>String str = new String("droid");</code>
這兩種方式我們在代碼編寫時都經常使用,尤其是字面量的方式。然而這兩種實作其實存在着一些性能和記憶體占用的差别。這一切都是源于JVM為了減少字元串對象的重複建立,其維護了一個特殊的記憶體,這段記憶體被成為字元串常量池或者字元串字面量池。
當我們使用了new來構造字元串對象的時候,不管字元串常量池中有沒有相同内容的對象,新的字元串對象都會建立(在堆中)。
堆:存放所有new出來的對象;
棧:存放基本類型的變量資料和對象的應用,對象(new出來的對象)本身并不存在棧中,而是存放在堆中或者常量池中(字元串常量對象存放在常量池中);
常量池:存放基本類型常量和字元串常量。
對于棧和常量池中的對象可以共享,對于堆中的對象不可以共享。棧中的資料大小和生命周期是可以确定的,當沒有引用指向資料時,這個資料就會自動消失。堆中的對象的由垃圾回收器負責回收,是以大小和生命周期不需要确定,具有很大的靈活性。
而對于字元串來說,其對象的引用都是存儲在棧中的,如果是編譯期已經建立好(即指用雙引号定義的)的就存儲在常量池中,如果是運作期(new出來的對象)則存儲在堆中。對于equals相等的字元串,在常量池中是隻有一份的,在堆中則有多份。
兩個深入閱讀的連結
字元串常量池: http://droidyue.com/blog/2014/12/21/string-literal-pool-in-java/
字元串拼接内部實作:http://droidyue.com/blog/2014/08/30/java-details-string-concatenation/