StringBuffer(StringBuilder)類和String一樣,也用來代表字符串,只是由於StringBuffer的內部實現方式和String不同,所以StringBuffer在進行字符串處理時,不生成新的對象,在內存使用上要優於String類。
所以在實際使用時,如果經常需要對一個字符串進行修改,例如插入、刪除等操作,使用StringBuffer要更加適合一些。
在StringBuffer類中存在很多和String類一樣的方法,這些方法在功能上和String類中的功能是完全一樣的。
但是有一個最顯著的區別在於,對於StringBuffer對象的每次修改都會改變對象自身,這點是和String類最大的區別。
另外由於StringBuffer是線程安全的,關於線程的概念後續有專門的章節進行介紹,所以在多線程程序中也可以很方便的進行使用,但是程序的執行效率相對來說就要稍微慢一些。
1. String 類
String的值是不可變的,這就導致每次對String的操作都會生成新的String對象,不僅效率低下,而且大量浪費有限的內存空間。
String a = "a"; //假設a指向地址0x0001
a = "b";//重新賦值後a指向地址0x0002,但0x0001地址中保存的"a"依舊存在,但已經不再是a所指向的,a 已經指向了其它地址。
因此String的操作都是改變賦值地址而不是改變值操作。
2.StringBuffer(StringBuilder)類
StringBuffer是可變類,和線程安全的字符串操作類,任何對它指向的字符串的操作都不會產生新的對象。 每個StringBuffer對象都有一定的緩衝區容量,當字符串大小沒有超過容量時,不會分配新的容量,當字符串大小超過容量時,會自動增加容量。
StringBuffer buf=new StringBuffer(); //分配長16字節的字符緩衝區
StringBuffer buf=new StringBuffer(512); //分配長512字節的字符緩衝區
StringBuffer buf=new StringBuffer("this is a test")//在緩衝區中存放了字符串,並在後面預留了16字節的空緩衝區。
1、StringBuffer對象的初始化
StringBuffer對象的初始化不像String類的初始化一樣,Java提供的有特殊的語法,而通常情況下一般使用構造方法進行初始化。
例如:
StringBuffer s = new StringBuffer();
這樣初始化出的StringBuffer對象是一個空的對象。
如果需要創建帶有內容的StringBuffer對象,則可以使用:
StringBuffer s = new StringBuffer(“abc”);
這樣初始化出的StringBuffer對象的內容就是字符串”abc”。
需要注意的是,StringBuffer和String屬於不同的類型,也不能直接進行強制類型轉換,下面的代碼都是錯誤的:
StringBuffer s = “abc”; //賦值類型不匹配
StringBuffer s = (StringBuffer)”abc”; //不存在繼承關係,無法進行強轉
StringBuffer對象和String對象之間的互轉的代碼如下:
String s = “abc”;
StringBuffer sb1 = new StringBuffer(“123”);
StringBuffer sb2 = new StringBuffer(s); //String轉換爲StringBuffer
String s1 = sb1.toString(); //StringBuffer轉換爲String
2、StringBuffer的常用方法
StringBuffer類中的方法主要偏重於對於字符串的變化,例如追加、插入和刪除等,這個也是StringBuffer和String類的主要區別。
a、append方法
public StringBuffer append(boolean b)
該方法的作用是追加內容到當前StringBuffer對象的末尾,類似於字符串的連接。調用該方法以後,StringBuffer對象的內容也發生改變,例如:
StringBuffer sb = new StringBuffer(“abc”);
sb.append(true);
則對象sb的值將變成”abctrue”。
使用該方法進行字符串的連接,將比String更加節約內容,例如應用於數據庫SQL語句的連接,例如:
StringBuffer sb = new StringBuffer();
String user = “test”;
String pwd = “123”;
sb.append(“select * from userInfo where username=“)
.append(user)
.append(“ and pwd=”)
.append(pwd);
這樣對象sb的值就是字符串“select * from userInfo where username=test and pwd=123”。
b、deleteCharAt方法
public StringBuffer deleteCharAt(int index)
該方法的作用是刪除指定位置的字符,然後將剩餘的內容形成新的字符串。例如:
StringBuffer sb = new StringBuffer(“Test”);
sb. deleteCharAt(1);
該代碼的作用刪除字符串對象sb中索引值爲1的字符,也就是刪除第二個字符,剩餘的內容組成一個新的字符串。所以對象sb的值變爲”Tst”。
還存在一個功能類似的delete方法:
public StringBuffer delete(int start,int end)
該方法的作用是刪除指定區間以內的所有字符,包含start,不包含end索引值的區間。例如:
StringBuffer sb = new StringBuffer(“TestString”);
sb. delete (1,4);
該代碼的作用是刪除索引值1(包括)到索引值4(不包括)之間的所有字符,剩餘的字符形成新的字符串。則對象sb的值是”TString”。
c、insert方法
public StringBuffer insert(int offset, boolean b)
該方法的作用是在StringBuffer對象中插入內容,然後形成新的字符串。例如:
StringBuffer sb = new StringBuffer(“TestString”);
sb.insert(4,false);
該示例代碼的作用是在對象sb的索引值4的位置插入false值,形成新的字符串,則執行以後對象sb的值是”TestfalseString”。
d、reverse方法
public StringBuffer reverse()
該方法的作用是將StringBuffer對象中的內容反轉,然後形成新的字符串。例如:
StringBuffer sb = new StringBuffer(“abc”);
sb.reverse();
經過反轉以後,對象sb中的內容將變爲”cba”。
e、setCharAt方法
public void setCharAt(int index, char ch)
該方法的作用是修改對象中索引值爲index位置的字符爲新的字符ch。例如:
StringBuffer sb = new StringBuffer(“abc”);
sb.setCharAt(1,’D’);
則對象sb的值將變成”aDc”。
f、trimToSize方法
public void trimToSize()
該方法的作用是將StringBuffer對象的中存儲空間縮小到和字符串長度一樣的長度,減少空間的浪費。
總之,在實際使用時,String和StringBuffer各有優勢和不足,可以根據具體的使用環境,選擇對應的類型進行使用。
最後是三者的耗時比較:
long currentTime, lastTime;
public void testString() {
String str = "";
String test = "";
int times = 10000;
StringBuffer buffer = new StringBuffer();
StringBuilder builder = new StringBuilder();
currentTime = System.currentTimeMillis();
for (int i = 0; i < times; i++) {
str += i;
}
lastTime = System.currentTimeMillis();
System.err.println("MS=" + (lastTime - currentTime));
currentTime = System.currentTimeMillis();
for (int i = 0; i < times; i++) {
buffer.append(i + "");
//此部分只在需要轉爲字符串的時候使用
// if (i == 999) {
// test = buffer.toString();
// }
}
// System.err.println("111111=" + test);
lastTime = System.currentTimeMillis();
System.err.println("MS1=" + (lastTime - currentTime));
currentTime = System.currentTimeMillis();
for (int i = 0; i < times; i++) {
builder.append(i + "");
}
lastTime = System.currentTimeMillis();
System.err.println("MS2=" + (lastTime - currentTime));
}
得到相同結果耗費的時間(毫秒)分別:
MS=586
MS1=6
MS2=2