簡單介紹
java中用於處理字符串常用的有三個類:
1、java.lang.String
2、java.lang.StringBuffer
3、java.lang.StrungBuilder
三者共同之處:都是final類,不允許被繼承,主要是從性能和安全性上考慮的,因為這幾個類都是經常被使用着,且考慮到防止其中的參數被參數修改影響到其他的應用。
StringBuffer是線程安全,可以不需要額外的同步用於多線程中;
StringBuilder是非同步,運行於多線程中就需要使用着單獨同步處理,但是速度就比StringBuffer快多了;
StringBuffer與StringBuilder兩者共同之處:可以通過append、indert進行字符串的操作。
String實現了三個接口:Serializable、Comparable<String>、CarSequence
StringBuilder只實現了兩個接口Serializable、CharSequence,相比之下String的實例可以通過compareTo方法進行比較,其他兩個不可以。
這三個類之間的區別主要是在兩個方面,即運行速度和線程安全這兩方面。
1、首先說運行速度,或者說是執行速度,在這方面運行速度快慢為:StringBuilder > StringBuffer > String
String最慢的原因:String為字符串常量,而StringBuilder和StringBuffer均為字符串變量,即String對象一旦創建之后該對象是不可更改的,但后兩者的對象是變量,是可以更改的。以下面一段代碼為例:
1 String str="abc"; 2 System.out.println(str); 3 str=str+"de"; 4 System.out.println(str);
運行這段代碼會發現先輸出“abc”,然后又輸出“abcde”,好像是str這個對象被更改了,其實,這只是一種假象罷了,JVM對於這幾行代碼是這樣處理的,首先創建一個String對象str,並把“abc”賦值給str,然后在第三行中,其實JVM又創建了一個新的對象也名為str,然后再把原來的str的值和“de”加起來再賦值給新的str,而原來的str就會被JVM的垃圾回收機制(GC)給回收掉了,所以,str實際上並沒有被更改,也就是前面說的String對象一旦創建之后就不可更改了。所以,Java中對String對象進行的操作實際上是一個不斷創建新的對象並且將舊的對象回收的一個過程,所以執行速度很慢。
而StringBuilder和StringBuffer的對象是變量,對變量進行操作就是直接對該對象進行更改,而不進行創建和回收的操作,所以速度要比String快很多。
另外,有時候我們會這樣對字符串進行賦值
1 String str="abc"+"de"; 2 StringBuilder stringBuilder=new StringBuilder().append("abc").append("de"); 3 System.out.println(str); 4 System.out.println(stringBuilder.toString());
這樣輸出結果也是“abcde”和“abcde”,但是String的速度卻比StringBuilder的反應速度要快很多,這是因為第1行中的操作和String str="abcde";是完全一樣的,所以會很快,而如果寫成下面這種形式
1 String str1="abc"; 2 String str2="de"; 3 String str=str1+str2;
那么JVM就會像上面說的那樣,不斷的創建、回收對象來進行這個操作了。速度就會很慢。
public static void main(String[] args) { long a=new Date().getTime(); String cc=""; int n=10000; for (int i = 0; i < n; i++) { cc+="."+i; } System.out.println("String使用的時間"+(System.currentTimeMillis()-a)/1000.0+"s"); long s1=System.currentTimeMillis(); StringBuilder sb=new StringBuilder(); for (int i = 0; i < n; i++) { sb.append("."+i); } System.out.println("StringBuilder使用的時間"+(System.currentTimeMillis()-s1)/1000.0+"s"); long s2=System.currentTimeMillis(); StringBuffer sbf=new StringBuffer(); for (int i = 0; i < n; i++) { sbf.append("."+i); } System.out.println("StringBuffer使用的時間"+(System.currentTimeMillis()-s2)/1000.0+"s"); }
2. 再來說線程安全
在線程安全上,StringBuilder是線程不安全的,而StringBuffer是線程安全的
如果一個StringBuffer對象在字符串緩沖區被多個線程使用時,StringBuffer中很多方法可以帶有synchronized關鍵字,所以可以保證線程是安全的,但StringBuilder的方法則沒有該關鍵字,所以不能保證線程安全,有可能會出現一些錯誤的操作。所以如果要進行的操作是多線程的,那么就要使用StringBuffer,但是在單線程的情況下,還是建議使用速度比較快的StringBuilder。
(一個線程訪問一個對象中的synchronized(this)同步代碼塊時,其他試圖訪問該對象的線程將被阻塞)
3. 總結一下
String:適用於少量的字符串操作的情況
StringBuilder:適用於單線程下在字符緩沖區進行大量操作的情況
StringBuffer:適用多線程下在字符緩沖區進行大量操作的情況