Java之String、StringBuffer、StringBuilder的區(qū)別分析
相信大家對 String 和 StringBuffer 的區(qū)別也已經(jīng)很了解了,但是估計還是會有很多同志對這兩個類的工作原理有些不清楚的地方,今天我在這里重新把這個概念給大家復(fù)習一下,順便牽出 J2SE 5.0 里面帶來的一個新的字符操作的類—— StringBuilder 。那么這個 StringBuilder 和 StringBuffer 以及我們最早遇見的 String 類有那些區(qū)別呢?在不同的場合下我們應(yīng)該用哪個呢?我講講自己對這幾個類的一點看法,也希望大家提出意見,每個人都有錯的地方,在錯了改的同時更是一個學習的好機會。
簡要的說, String 類型和 StringBuffer 類型的主要性能區(qū)別其實在于 String 是不可變的對象(為什么?問問 Java 的設(shè)計者吧,為什么 String 不是原生類型呢?)因此在每次對 String 類型進行改變的時候其實都等同于生成了一個新的 String 對象,然后將指針指向新的 String 對象,所以經(jīng)常改變內(nèi)容的字符串最好不要用 String ,因為每次生成對象都會對系統(tǒng)性能產(chǎn)生影響,特別當內(nèi)存中無引用對象多了以后, JVM 的 GC 就會開始工作,那速度是一定會相當慢的。這里嘗試舉個不是很恰當?shù)睦樱?br />
String S1 = "abc";
For(int I = 0 ; I < 10000 ; I ++) // For 模擬程序的多次調(diào)用
{
S1 + = "def";
S1 = "abc";
}
如果是這樣的話,到這個 for 循環(huán)完畢后,如果內(nèi)存中的對象沒有被 GC 清理掉的話,內(nèi)存中一共有 2 萬多個了,驚人的數(shù)目,而如果這是一個很多人使用的系統(tǒng),這樣的數(shù)目就不算很多了,所以大家使用的時候一定要小心。
而如果是使用 StringBuffer 類則結(jié)果就不一樣了,每次結(jié)果都會對 StringBuffer 對象本身進行操作,而不是生成新的對象,再改變對象引用。所以在一般情況下我們推薦使用 StringBuffer ,特別是字符串對象經(jīng)常改變的情況下。而在某些特別情況下, String 對象的字符串拼接其實是被 JVM 解釋成了 StringBuffer 對象的拼接,所以這些時候 String 對象的速度并不會比 StringBuffer 對象慢,而特別是以下的字符串對象生成中, String 效率是遠要比 StringBuffer 快的:
String S1 = "This is only a" + " simple" + " test";
StringBuffer Sb = new StringBuilder("This is only a").append(" simple").append(" test");
你會很驚訝的發(fā)現(xiàn),生成 String S1 對象的速度簡直太快了,而這個時候 StringBuffer 居然速度上根本一點都不占優(yōu)勢。其實這是 JVM 的一個把戲,在 JVM 眼里,這個
String S1 = "This is only a" + " simple" + "test"; 其實就是: String S1 = "This is only a simple test"; 所以當然不需要太多的時間了。但大家這里要注意的是,如果你的字符串是來自另外的 String 對象的話,速度就沒那么快了,譬如:
String S2 = "This is only a";
String S3 = " simple";
String S4 = " test";
String S1 = S2 +S3 + S4;
這時候 JVM 會規(guī)規(guī)矩矩的按照原來的方式去做, S1 對象的生成速度就不像剛才那么快了,一會兒我們可以來個測試作個驗證。
由此我們得到第一步結(jié)論: 在大部分情況下 StringBuffer > String
而 StringBuilder 跟他們比又怎么樣呢?先簡單介紹一下, StringBuilder 是 JDK5.0 中新增加的一個類,它跟 StringBuffer 的區(qū)別看下面的介紹(來源 JavaWorld ):
Java.lang.StringBuffer 線程安全的可變字符序列。類似于 String 的字符串緩沖區(qū),但不能修改??蓪⒆址彌_區(qū)安全地用于多個線程。可以在必要時對這些方法進行同步,因此任意特定實例上的所有操作就好像是以串行順序發(fā)生的,該順序與所涉及的每個線程進行的方法調(diào)用順序一致。
每個字符串緩沖區(qū)都有一定的容量。只要字符串緩沖區(qū)所包含的字符序列的長度沒有超出此容量,就無需分配新的內(nèi)部緩沖區(qū)數(shù)組。如果內(nèi)部緩沖區(qū)溢出,則此容量自動增大。從 JDK 5.0 開始,為該類增添了一個單個線程使用的等價類,即 StringBuilder 。與該類相比,通常應(yīng)該優(yōu)先使用 StringBuilder 類,因為它支持所有相同的操作,但由于它不執(zhí)行同步,所以速度更快。
但是如果將 StringBuilder 的實例用于多個線程是不安全的。需要這樣的同步,則建議使用 StringBuffer 。
這樣說估計大家都能明白他們之間的區(qū)別了,那么下面我們再做一個一般性推導(dǎo):
在大部分情況下 StringBuilder > StringBuffer
因此,根據(jù)這個不等式的傳遞定理: 在大部分情況下 StringBuilder > StringBuffer > String
既然有這樣的推導(dǎo)結(jié)果了,我們做個測試驗證一下:
測試代碼如下:
public class testssb {
/** Creates a new instance of testssb */
final static int ttime = 10000;// 測試循環(huán)次數(shù)
public testssb() {
}
public void test(String s){
long begin = System.currentTimeMillis();
for(int i=0;i<ttime;i++){
s += "add";
}
long over = System.currentTimeMillis();
System.out.println(" 操作 "+s.getClass().getName()+" 類型使用的時間為: " + (over - begin) + " 毫秒 " );
}
public void test(StringBuffer s){
long begin = System.currentTimeMillis();
for(int i=0;i<ttime;i++){
s.append("add");
}
long over = System.currentTimeMillis();
System.out.println(" 操作 "+s.getClass().getName()+" 類型使用的時間為: " + (over - begin) + " 毫秒 " );
}
public void test(StringBuilder s){
long begin = System.currentTimeMillis();
for(int i=0;i<ttime;i++){
s.append("add");
}
long over = System.currentTimeMillis();
System.out.println(" 操作 "+s.getClass().getName()+" 類型使用的時間為: " + (over - begin) + " 毫秒 " );
}
// 對 String 直接進行字符串拼接的測試
public void test2(){
String s2 = "abadf";
long begin = System.currentTimeMillis();
for(int i=0;i<ttime;i++){
String s = s2 + s2 + s2 ;
}
long over = System.currentTimeMillis();
System.out.println(" 操作字符串對象引用相加類型使用的時間為: " + (over - begin) + " 毫秒 " );
}
public void test3(){
long begin = System.currentTimeMillis();
for(int i=0;i<ttime;i++){
String s = "abadf" + "abadf" + "abadf" ;
}
long over = System.currentTimeMillis();
System.out.println(" 操作字符串相加使用的時間為: "+ (over - begin) + " 毫秒 " );
}
public static void main(String[] args){
String s1 ="abc";
StringBuffer sb1 = new StringBuffer("abc");
StringBuilder sb2 = new StringBuilder("abc");
testssb t = new testssb();
t.test(s1);
t.test(sb1);
t.test(sb2);
t.test2();
t.test3();
}
}
以上代碼在 NetBeans 5.0 IDE/JDK1.6 上編譯通過,循環(huán)次數(shù) ttime 為 10000 次的測試結(jié)果如下:
操作 java.lang.String 類型使用的時間為: 4392 毫秒
操作 java.lang.StringBuffer 類型使用的時間為: 0 毫秒
操作 java.lang.StringBuilder 類型使用的時間為: 0 毫秒
操作字符串對象引用相加類型使用的時間為: 15 毫秒
操作字符串相加使用的時間為: 0 毫秒
好像還看不出 StringBuffer 和 StringBuilder 的區(qū)別,把 ttime 加到 30000 次看看:
操作 java.lang.String 類型使用的時間為: 53444 毫秒
操作 java.lang.StringBuffer 類型使用的時間為: 15 毫秒
操作 java.lang.StringBuilder 類型使用的時間為: 15 毫秒
操作字符串對象引用相加類型使用的時間為: 31 毫秒
操作字符串相加使用的時間為: 0 毫秒
StringBuffer 和 StringBuilder 的性能上還是沒有太大的差異,再加大到 100000 看看,這里就不加入對 String 類型的測試了,因為對 String 類型這么大數(shù)據(jù)量的測試會很慢滴……
操作 java.lang.StringBuffer 類型使用的時間為: 31 毫秒
操作 java.lang.StringBuilder 類型使用的時間為: 16 毫秒
能看出差別了,但其中有多次的測試結(jié)果居然是 StringBuffer 比 StringBuilder 快,再加大一些到 1000000 看看:
操作 java.lang.StringBuffer 類型使用的時間為: 265 毫秒
操作 java.lang.StringBuilder 類型使用的時間為: 219 毫秒
有些少區(qū)別了,而且結(jié)果很穩(wěn)定,再大點看看, ttime = 5000000 :
······ Exception in thread "main" java.lang.OutOfMemoryError: Java heap space ······
呵呵,算了,不去測試了,基本來說都是在性能上都是 StringBuilder > StringBuffer > String 的了。
相關(guān)文章
java+sqlserver實現(xiàn)學生信息管理系統(tǒng)
這篇文章主要介紹了利用java和sqlserver實現(xiàn)學生信息管理系統(tǒng),違章內(nèi)容主要建立了與sqlserver數(shù)據(jù)庫的連接開始展開內(nèi)容,能學到了解JDBC執(zhí)行SQL的語法,需要的朋友可以參考一下2021-12-12
Java縮略圖生成庫之Thumbnailator應(yīng)用說明
Thumbnailator是一個為Java界面更流暢的縮略圖生成庫,從API提供現(xiàn)有的圖像文件和圖像對象的縮略圖中簡化了縮略過程,兩三行代碼就能夠從現(xiàn)有圖片生成縮略圖,使用起來非常方便,需要的朋友可以了解下2012-12-12
SpringBoot運用Redis統(tǒng)計用戶在線數(shù)量的兩種方法實現(xiàn)
本文主要介紹了SpringBoot運用Redis統(tǒng)計用戶在線數(shù)量的兩種方法實現(xiàn),包括通過RedisSet精確記錄用戶狀態(tài),或用RedisBitmap按位存儲優(yōu)化內(nèi)存,Set適合小規(guī)模場景,Bitmap適用于大規(guī)模連續(xù)ID,可根據(jù)需求選擇實現(xiàn)方式2025-06-06
SpringCloud?openfeign聲明式服務(wù)調(diào)用實現(xiàn)方法介紹
在springcloud中,openfeign是取代了feign作為負載均衡組件的,feign最早是netflix提供的,他是一個輕量級的支持RESTful的http服務(wù)調(diào)用框架,內(nèi)置了ribbon,而ribbon可以提供負載均衡機制,因此feign可以作為一個負載均衡的遠程服務(wù)調(diào)用框架使用2022-12-12
Spring Security實現(xiàn)5次密碼錯誤觸發(fā)賬號自動鎖定功能
在現(xiàn)代互聯(lián)網(wǎng)應(yīng)用中,賬號安全是重中之重,然而,暴力 破解攻擊依然是最常見的安全威脅之一,攻擊者通過自動化腳本嘗試大量的用戶名和密碼組合,試圖找到漏洞進入系統(tǒng),所以為了解決這一問題,賬號鎖定機制被廣泛應(yīng)用,本文介紹了Spring Security實現(xiàn)5次密碼錯誤觸發(fā)賬號鎖定功能2024-12-12
SpringMVC4.3解析器HandlerMethodArgumentResolver接口源碼
這篇文章主要為大家介紹了SpringMVC4.3解析器HandlerMethodArgumentResolver接口源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-09-09
spring+maven實現(xiàn)發(fā)送郵件功能
這篇文章主要為大家詳細介紹了spring+maven實現(xiàn)發(fā)送郵件功能,利用spring提供的郵件工具來發(fā)送郵件,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-07-07

