java中String、StringBuffer、StringBuilder的區別java
java中String、StringBuffer、StringBuilder是編程中常常使用的字符串類,他們之間的區別也是常常在面試中會問到的問題。如今總結一下,看看他們的不一樣與相同。面試
1.可變與不可變編程
String類中使用字符數組保存字符串,以下就是,由於有「final」修飾符,因此能夠知道string對象是不可變的。數組
private final char value[];安全
StringBuilder與StringBuffer都繼承自AbstractStringBuilder類,在AbstractStringBuilder中也是使用字符數組保存字符串,以下就是,可知這兩種對象都是可變的。多線程
char[] value;app
2.是否多線程安全ide
String中的對象是不可變的,也就能夠理解爲常量,顯然線程安全。ui
AbstractStringBuilder是StringBuilder與StringBuffer的公共父類,定義了一些字符串的基本操做,如expandCapacity、append、insert、indexOf等公共方法。this
StringBuffer對方法加了同步鎖或者對調用的方法加了同步鎖,因此是線程安全的。看以下源碼:
1 public synchronized StringBuffer reverse() {2 super.reverse();3 return this;4 }5 6 public int indexOf(String str) {7 return indexOf(str, 0); //存在 public synchronized int indexOf(String str, int fromIndex) 方法8 }
StringBuilder並無對方法進行加同步鎖,因此是非線程安全的。
3.StringBuilder與StringBuffer共同點
StringBuilder與StringBuffer有公共父類AbstractStringBuilder(抽象類)。
抽象類與接口的其中一個區別是:抽象類中能夠定義一些子類的公共方法,子類只須要增長新的功能,不須要重複寫已經存在的方法;而接口中只是對方法的申明和常量的定義。
StringBuilder、StringBuffer的方法都會調用AbstractStringBuilder中的公共方法,如super.append(...)。只是StringBuffer會在方法上加synchronized關鍵字,進行同步。
若是程序不是多線程的,那麼使用StringBuilder效率高於StringBuffer。
4.三者在執行速度方面的比較:StringBuilder > StringBuffer > String
5.String <(StringBuffer,StringBuilder)的緣由
String:字符串常量
StringBuffer:字符創變量
StringBuilder:字符創變量
從上面的名字能夠看到,String是「字符創常量」,也就是不可改變的對象。對於這句話的理解你可能會產生這樣一個疑問 ,好比這段代碼:
1 String s = "abcd";
2 s = s+1;
3 System.out.print(s);// result : abcd1
咱們明明就是改變了String型的變量s的,爲何說是沒有改變呢? 其實這是一種欺騙,JVM是這樣解析這段代碼的:首先建立對象s,賦予一個abcd,而後再建立一個新的對象s用來 執行第二行代碼,也就是說咱們 以前對象s並無變化,因此咱們說String類型是不可改變的對象了,因爲這種機制,每當用String操做字符串時,其實是在不斷的建立新的對象, 而原來的對象就會變爲垃圾被GC回收掉,可想而知這樣執行效率會有多底。
而StringBuffer與StringBuilder就不同了,他們是字符串變量,是可改變的對象,每當咱們用它們對字符串作操做時,其實是在一個對象上操做的,這樣就不會像String同樣建立一些而外的對象進行操做了,固然速度就快了。
6.一個特殊的例子:
1 String str = 「This is only a」 + 「 simple」 + 「 test」;
3 StringBuffer builder = new StringBuilder(「This is only a」).append(「 simple」).append(「 test」);
你會很驚訝的發現,生成str對象的速度簡直太快了,而這個時候StringBuffer竟然速度上根本一點都不佔優點。其實這是JVM的一個把戲,實際上:
String str = 「This is only a」 + 「 simple」 + 「test」;
其實就是:
String str = 「This is only a simple test」;
因此不須要太多的時間了。但你們這裏要注意的是,若是你的字符串是來自另外的String對象的話,速度就沒那麼快了,譬如:
String str2 = 「This is only a」;
String str3 = 「 simple」;
String str4 = 「 test」;
String str1 = str2 +str3 + str4;
這時候JVM會規規矩矩的按照原來的方式去作。
7.StringBuilder與 StringBuffer
StringBuilder:線程非安全的
StringBuffer:線程安全的
當咱們在字符串緩衝去被多個線程使用是,JVM不能保證StringBuilder的操做是安全的,雖然他的速度最快,可是能夠保證 StringBuffer是能夠正確操做的。固然大多數狀況下就是咱們是在單線程下進行的操做,因此大多數狀況下是建議用StringBuilder而不 用StringBuffer的,就是速度的緣由。
對於三者使用的總結: 1.若是要操做少許的數據用 = String
2.單線程操做字符串緩衝區 下操做大量數據 = StringBuilder
3.多線程操做字符串緩衝區 下操做大量數據 = StringBuffer