String、StringBuffer和StringBuilder的區別

String
  String類是不可變類,即一旦一個String對象被創建以後,包含在這個對象中的字符序列是不可改變的,直至這個對象被銷燬。

  這個是String類的解釋,之前小鹹兒看到這個情況,不能理解上述的解釋,如下

String a = "123";
a = "456";
// 打印出來的a爲456
System.out.println(a)
1
2
3
4


  看到這裏,小鹹兒不明白了,這不是明明已經對他進行修改了嗎?爲什麼還說他是一個不可變類呢?

  經過小鹹兒和小夥伴們的學習,明白String類不可變在哪裏體現出來的,接下來就看一張上述a對象的內存存儲空間圖


  可以看出來,再次給a賦值時,並不是對原來堆中實例對象進行重新賦值,而是生成一個新的實例對象,並且指向“456”這個字符串,a則指向最新生成的實例對象,之前的實例對象仍然存在,如果沒有被再次引用,則會被垃圾回收。

StringBuffer
  StringBuffer對象則代表一個字符序列可變的字符串,當一個StringBuffer被創建以後,通過StringBuffer提供的append()、insert()、reverse()、setCharAt()、setLength()等方法可以改變這個字符串對象的字符序列。一旦通過StringBuffer生成了最終想要的字符串,就可以調用它的toString()方法將其轉換爲一個String對象。

StringBuffer b = new StringBuffer("123");
b.append("456");
// b打印結果爲:123456
System.out.println(b);
1
2
3
4


  在看一下b對象的內存空間圖:


  所以說StringBuffer對象是一個字符序列可變的字符串,它沒有重新生成一個對象,而且在原來的對象中可以連接新的字符串。

StringBuilder
  StringBuilder類也代表可變字符串對象。實際上,StringBuilder和StringBuffer基本相似,兩個類的構造器和方法也基本相同。不同的是:StringBuffer是線程安全的,而StringBuilder則沒有實現線程安全功能,所以性能略高。

StringBuffer是如何實現線程安全的呢?
StringBuffer類中實現的方法:

StringBuilder類中實現的方法:

  由此可見,StringBuffer類中的方法都添加了synchronized關鍵字,也就是給這個方法添加了一個鎖,用來保證線程安全。

Java9的改進
  Java9改進了字符串(包括String、StringBuffer、StringBuilder)的實現。在Java9以前字符串採用char[]數組來保存字符,因此字符串的每個字符佔2字節;而Java9的字符串採用byte[]數組再加一個encoding-flag字段來保存字符,因此字符串的每個字符只佔1字節。所以Java9的字符串更加節省空間,字符串的功能方法也沒有受到影響。
 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章