從運(yùn)行速度和線程安全這兩方面來看一下三者之間的區(qū)別鹏溯。
運(yùn)行速度
StringBuilder > StringBuffer > String
String最慢的原因:
String為字符串常量罢维,而StringBuilder和StringBuffer均為字符串變量,即String對象一旦創(chuàng)建之后該對象是不可更改的剿涮,但后兩者的對象是變量言津,是可以更改的攻人。
String str = "ABC";
System.out.println(str);
str = str + "DEF";
System.out.println(str);
如果運(yùn)行這段代碼會(huì)發(fā)現(xiàn)先輸出“ABC”取试,然后又輸出“ABCDEF”,好像是str這個(gè)對象被更改了怀吻,其實(shí)瞬浓,這只是一種假象罷了,JVM對于這幾行代碼是這樣處理的蓬坡,首先創(chuàng)建一個(gè)String對象str猿棉,并把“ABC”賦值給str磅叛,然后在第三行中,其實(shí)JVM又創(chuàng)建了一個(gè)新的對象也名為str萨赁,然后再把原來的str的值和“DEF”加起來再賦值給新的str弊琴,而原來的str就會(huì)被JVM的垃圾回收機(jī)制(GC)給回收掉了,所以杖爽,str實(shí)際上并沒有被更改敲董,也就是前面說的String對象一旦創(chuàng)建之后就不可更改了。所以慰安,Java中對String對象進(jìn)行的操作實(shí)際上是一個(gè)不斷創(chuàng)建新的對象并且將舊的對象回收的一個(gè)過程腋寨,所以執(zhí)行速度很慢。而StringBuilder和StringBuffer的對象是變量化焕,對變量進(jìn)行操作就是直接對該對象進(jìn)行更改萄窜,而不進(jìn)行創(chuàng)建和回收的操作,所以速度要比String快很多撒桨。
有時(shí)候我們會(huì)這樣對字符串進(jìn)行賦值
String str="ABC"+"DEF";
StringBuilder stringBuilder=new StringBuilder().append("ABC").append("DEF");
System.out.println(str);
System.out.println(stringBuilder.toString());
這樣輸出結(jié)果也是“ABCDEF”和“ABCDEF”查刻,但是String的速度卻比StringBuilder的反應(yīng)速度要快很多,這是因?yàn)榈?行中的操作和
String str="ABCDEF";
是完全一樣的凤类,所以會(huì)很快赖阻,而如果寫成下面這種形式
String str1="ABC";
String str2="DEF";
String str = str1 + str2;
那么JVM就會(huì)像上面說的那樣,不斷的創(chuàng)建踱蠢、回收對象來進(jìn)行這個(gè)操作了火欧。速度就會(huì)很慢。
線程安全
在線程安全上茎截,StringBuilder是線程不安全的苇侵,而StringBuffer是線程安全的
如果一個(gè)StringBuffer對象在字符串緩沖區(qū)被多個(gè)線程使用時(shí),StringBuffer中很多方法可以帶有synchronized關(guān)鍵字企锌,所以可以保證線程是安全的榆浓,但StringBuilder的方法則沒有該關(guān)鍵字,所以不能保證線程安全撕攒,有可能會(huì)出現(xiàn)一些錯(cuò)誤的操作陡鹃。所以如果要進(jìn)行的操作是多線程的,那么就要使用StringBuffer抖坪,但是在單線程的情況下萍鲸,還是建議使用速度比較快的StringBuilder。
總結(jié)
String:適用于少量的字符串操作的情況
StringBuilder:適用于單線程下在字符緩沖區(qū)進(jìn)行大量操作的情況
StringBuffer:適用多線程下在字符緩沖區(qū)進(jìn)行大量操作的情況
String和StringBuffer轉(zhuǎn)化
public static void main(String[] args){
// String->StringBuffer
String str = "Hello";
StringBuffer sb = new StringBuffer(str);
System.out.println(sb.toString());
// StringBuffer->String
StringBuffer buffer = new StringBuffer("java");
String s = buffer.toString();
System.out.println(s);
}