從內(nèi)存方面解釋Java中String與StringBuilder的性能差異
以前經(jīng)常在網(wǎng)上看到關(guān)于Java字符串拼接等方面的討論??吹接行㎎ava開發(fā)人員在給新手程序員的建議中類似如下寫道:
不要使用+號拼接字符串,要使用StringBuffer或StringBuilder的append()方法來拼接字符串。
不過,用+號拼接字符串就真的那么令人討厭,難道使用+號拼接字符串就沒有一點可取之處嗎?
通過查閱Java API文檔中關(guān)于String類的部分內(nèi)容,我們可以看到如下片段:
“Java 語言提供對字符串串聯(lián)符號("+")以及將其他對象轉(zhuǎn)換為字符串的特殊支持。字符串串聯(lián)是通過 StringBuilder(或 StringBuffer)類及其 append 方法實現(xiàn)的。字符串轉(zhuǎn)換是通過 toString 方法實現(xiàn)的,該方法由 Object 類定義,并可被 Java中的所有類繼承?!?/p>
這段話很明確地告訴我們,在Java中使用+號拼接字符串,實際上使用的就是StringBuffer或StringBuilder及其append方法來實現(xiàn)的。
除了Java API文檔,我們還可以使用工具查看class類文件的字節(jié)碼命令來得到上述答案。 例如代碼:
public static void main(String[] args) {
String a = "Hello";
String b = " world";
String str = a + b + " !";
System.out.println(str);
}
通過工具查看到其對應(yīng)的字節(jié)碼命令如下:

從字節(jié)碼命令中,我們可以清楚地看到,我們編寫的如下代碼
String str = a + b + " !";
被編譯器轉(zhuǎn)換成了類似如下語句:
String str = new StringBuilder(String.valueOf(a)).append(b).append(" !").toString();
不僅如此,Java的編譯器也是一個比較聰明的編譯器,當(dāng)+號拼接的全部是字符串字面量時,Java的編譯器將會在編譯時智能地將其轉(zhuǎn)換為一個完整的字符串。例如:
public static void main(String[] args) {
String str = "Hello" + " world" + ", Java!";
System.out.println(str);
}
Java編譯器直接將這種全是字面量的字符串拼接,在編譯時就轉(zhuǎn)換為了一個完整的字符串。

就算+號拼接的字符串中存在變量,Java編譯器也會將最前面的字符串字面量合并為一個字符串。
public static void main(String[] args) {
String java = ", Java!";
String str = "Hello" + " world" + java;
System.out.println(str);
}

從上述可知,對于類似String str = str1 + str2 + str3 + str4,這種將多個字符串一次性拼接的操作,使用+號來進行拼接是完全沒有問題的。
在Java中,String對象是不可變的(Immutable)。在代碼中,可以創(chuàng)建多個某一個String對象的別名。但是這些別名都是的引用是相同的。
比如s1和s2都是”droidyue.com”對象的別名,別名保存著到真實對象的引用。所以s1 = s2
String s1 = "droidyue.com";
String s2 = s1;
System.out.println("s1 and s2 has the same reference =" + (s1 == s2));
并且在Java中,唯一被重載的運算符就是字符串的拼接相關(guān)的。+,+=。除此之外,Java設(shè)計者不允許重載其他的運算符。
在Java中,唯一被重載的運算符就是字符串的拼接相關(guān)的。+,+=。除此之外,Java設(shè)計者不允許重載其他的運算符。
眾所周知,在Java 1.4版本之前,字符串拼接可以使用StringBuffer,從Java 1.5開始,我們可以使用StringBuilder來拼接字符串。StringBuffer和StringBuilder的主要區(qū)別在于:StringBuffer是線程安全的,適用于多線程操作字符串;StringBuilder是線程不安全的,適合單線程下操作字符串。不過,我們的大多數(shù)字符串拼接操作都是在單線程下進行的,因此使用StringBuilder有利于提高性能。
在Java 1.4之前,編譯器使用StringBuffer來處理+號拼接的字符串;從Java 1.5開始,編譯器大多數(shù)情況下都使用StringBuilder來處理+號拼接的字符串。
當(dāng)我們在JDK 1.4的環(huán)境下編寫代碼時,對于上述這種一次性拼接多個字符串的情況,建議最好使用+號來處理。這樣,當(dāng)JDK 升級到1.5及以上版本時,編譯器將會自動將其轉(zhuǎn)換為StringBuilder來拼接字符串,從而提高字符串拼接效率。
當(dāng)然,推薦使用+號拼接字符串也僅限于在一條語句中拼接多個字符串時使用。如果分散在多條語句中拼接一個字符串,仍然建議使用StringBuffer或StringBuilder。 例如:
public static void main(String[] args) {
String java = ", Java!";
String str = "";
str += "Hello";
str += " world";
str += java;
System.out.println(str);
}
編譯器編譯后的字節(jié)命令如下:

從上面的圖片中我們可以知道,每一條+號拼接語句,都創(chuàng)建了一個新的StringBuilder對象。這種情況在循環(huán)條件下表現(xiàn)得尤其明顯,造成了相對較大的性能損耗。因此,在多條語句中拼接字符串,強烈建議使用StringBuffer或StringBuilder來處理。
關(guān)于使用StringBuilder帶來的優(yōu)化
此外,在使用StringBuffer或StringBuilder的時候,我們還可以使用如下方式,進一步提高性能(下面代碼以StringBuilder為例,StringBuffer與此類似)。
1.預(yù)測最終獲得的字符串的最大長度。
StringBuilder內(nèi)部char數(shù)組的默認長度為16,當(dāng)我們append追加字符串后超過此長度時,StringBuilder會擴大內(nèi)部的數(shù)組容量以滿足需要。在這個過程中,StringBuilder會創(chuàng)建一個新的較大容量的char數(shù)組,并將原數(shù)組中的數(shù)據(jù)復(fù)制到新數(shù)組中。如果我們能夠大致預(yù)測到最終拼接得到的字符串的最大長度,就可以在創(chuàng)建StringBuilder對象時指定合適大小的初始容量。例如,我們需要拼接獲得含有100個字母a的字符串。即可編寫如下代碼:
StringBuilder sb = new StringBuilder(100);
for (int i = 0; i < 100; i++) {
sb.append('a');
}
System.out.println(sb);
請根據(jù)實際情況進行平衡,以創(chuàng)建適合初始容量的StringBuilder。
2.對于單個字符,盡可能地使用char類型,而不是String類型。
有些時候,我們需要在字符串后追加單個字符(例如:a),此時應(yīng)盡可能地使用
sb.append('a');
而不是使用:
sb.append("a");
- 詳細分析Java中String、StringBuffer、StringBuilder類的性能
- 深入剖析java中String、StringBuffer、StringBuilder的區(qū)別
- 淺析java中stringBuilder的用法
- Java StringBuilder和StringBuffer源碼分析
- Java中String、StringBuffer、StringBuilder的區(qū)別介紹
- Java中StringBuffer和StringBuilder區(qū)別
- Java中的StringBuilder性能測試
- java中String與StringBuilder的區(qū)別
- 全面解釋java中StringBuilder、StringBuffer、String類之間的關(guān)系
- Java之String、StringBuffer、StringBuilder的區(qū)別分析
- Java中StringBuilder字符串類型的操作方法及API整理
相關(guān)文章
SpringMVC?Restful風(fēng)格與中文亂碼問題解決方案介紹
Restful就是一個資源定位及資源操作的風(fēng)格,不是標準也不是協(xié)議,只是一種風(fēng)格,是對http協(xié)議的詮釋,下面這篇文章主要給大家介紹了關(guān)于SpringMVC對Restful風(fēng)格支持的相關(guān)資料,需要的朋友可以參考下2022-10-10
Java實現(xiàn)簡易學(xué)籍管理系統(tǒng)
這篇文章主要為大家詳細介紹了Java實現(xiàn)簡易學(xué)籍管理系統(tǒng),文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-02-02
Mybatis在sqlite中無法讀寫byte[]類問題的解決辦法
這篇文章主要給大家介紹了關(guān)于Mybatis在sqlite中無法讀寫byte[]類問題的解決辦法,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-10-10
java request.getHeader("user-agent")獲取瀏覽器信息的方法
這篇文章主要介紹了java request.getHeader("user-agent")獲取瀏覽器信息的方法,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-03-03

