標籤:
我個人覺得理解String和StringBuffer以及StringBuilder的區別比較重要.在討論他們的區別時,我們首先應該知道java中的String.
首先討論String.
翻看String.java源碼,會知道string的一個重要秘密:
在string中,其實內部是通過一個char數組來維護這個string的,並且還定義了一個記錄該string在這個數組的起始位置的索引,以及定義了這個string的長度. 重點是他們都是final類型.
private final char[] value; private final int offset; private final int count;
我們知道final修飾的變數是不可更改的.所以很快不得不接受一個現實:java中每一個string其實都是不可以更改的. 所以如果需要對一個string內的字元變更/移動等操作很顯然是不可的. 這個時候會很快發現string中的方法replace()不是就可以更改一個string嗎?為了消除這個問號,我們就來看看replace其中一個源碼:
public String replace(char oldChar, char newChar) { char[] buffer = value; int _offset = offset; int _count = count; int idx = _offset; int last = _offset + _count; boolean copied = false; while (idx < last) { if (buffer[idx] == oldChar) { if (!copied) { char[] newBuffer = new char[_count]; System.arraycopy(buffer, _offset, newBuffer, 0, _count); buffer = newBuffer; idx -= _offset; last -= _offset; copied = true; } buffer[idx] = newChar; } idx++; } return copied ? new String(0, count, buffer) : this; }這個方法目的就是用newChar替換string中的所有的oldChar. 從上面的代碼可以清楚的知道:當替換成功時候返回的其實是一個新的string,也就是說並不是原來的那個string對象.
還包括substring在內的所以其他方法,貌似是在改這個string,其實都不是,最後都是new了一個新的string.
這樣一來問題就來了:往往實際開發中,對一個string反覆修改是再正常不過的事情了,而且有時候我們還會高頻率的這麼做,如果每次操作都是new一個新的string,顯然是不可的.在這種情況下StringBuffer和StringBuilder就出現了!目的就是為瞭解決這一問題.
查看StringBuffer和StringBuilder的源碼會很快發現:其實他們幾乎都一樣,唯一的區別就是StringBuffer的幾乎所有方法都採用了synchronized來同步,所以StringBuffer和StringBuilder區別就是安全執行緒問題.
所以接下來討論的焦點就是StringBuffer如何?對字串的修改的.
StringBuffer和StringBuilder都是實現了虛類:AbstractStringBuilder,其中關鍵功能的實現都在這個虛類中,在AbstractStringBuilder中用一個char數組來維護這個字串的,並且也定義了字串的長度,如下代碼所示:
private char[] value; private int count;
這個和string裡面定義的final char[] value不一樣. 這裡看看append()其中一個方法的實現:
final void append0(String string) { if (string == null) { appendNull(); return; } int length = string.length(); int newCount = count + length; if (newCount > value.length) { enlargeBuffer(newCount); } string._getChars(0, length, value, count); count = newCount; }上面的方法就是給StringBuffer或者StringBuilder後面追加一個string: 首先是確定當前的字串數組長度是否不夠,如果不夠就擴容,然後將需要追加的string通過方法拷貝到value中.
從中可以發現,由於StringBuffer裡面的字元數組value不是final,所以可以對其更改.而不需要像String那樣new一個新的String對象.
java中String和StringBuffer以及StringBuilder的區別