1. 程式人生 > >String String Buffer String Builder

String String Buffer String Builder

body 線程安全 start idt oid 新的 收起 wid als

Java參數傳遞

public class TEST{
float ptValue;

public void changeStr(String value){

value = new String("different");

}

public void changeObjValue(TEST ref){

ref.ptValue = 99.0f;

}

public static void main(String[] args){

String str;

Test pt = new TEST();

str = new String("Hello");

pt.changeStr(str);

System.out.println("Str value is: " + str );

pt.ptValue = 101.0f;

pt.changeObjValue(pt);

System.out.println("Pt value is: " + pt.ptValue );

}

不解:為什麽str的結果還是Hello。
String 不也是引用類型——類 嗎? 為什麽與傳一個對象TEST進去的結果(原來為101.0f,後來變為99.0f)不同?收起

///////////

Java的方法參數傳遞都是值傳遞。這個問題需要樓主慢慢理解,如果基礎薄弱(C、C++基礎弱)就記住就好。如果基礎好可以從內存角度理解。

從語法和簡單層面理解:str是一個String類型的對象,傳遞給changeStr方法,傳遞的是str的引用內容,方法內的changeStr(String value),value變量和str指向了同一個對象,但是value重新賦值後,它的指向改變了,但是str的指向不會發生,所以str的輸出還是“Hello”,而changeObjValue方法傳入的TEST的對象pt給changeObjValue(TEST ref),ref變量和pt變量指向了同一對象,之後通過改變了對象的內容,pt變量的指向沒變,但是執行的對象的變量變了。所以pt的ptValue值變了。

進一步理解,Java對字符串進行了特殊處理,因為字符串這種數據類型太常用了,為了節省內容提高效率。Java把字符串放入了常量池,所以我們沒辦法在運行時修改一個字符串對象的內容。也就是對value變量的賦值操作是重寫建立了一個對象。樓主可以測試經典的java String的代碼:

1

2

3

4

5

6

7

8

String a = "a";

String b = "a";

String c = new String("a");

String d = new String("a");

System.out.println(a==b);

System.out.println(a==c);

System.out.println(b==c);

System.out.println(d==c);

再深入,Java沒有C和C++的指針概念,可以粗暴的理解Java的引用是一種特殊的指針(實際上Java盡量把程序猿從指針中解放出來,或者說避免直接操作內存對象)。有一種不負責的說法:Java的基本類型傳遞傳遞的是值,也就是值的副本,引用類型傳遞的時引用。但是要註意String對象的內容不可變,而且傳遞的引用指向不可變。運行時Java的局部變量在棧空間中(線程私有),對象在堆內存中(線程公有),changeStr方法相當於在堆內存中又開辟了一塊兒空間重寫分配了一個對象,但是str沒有指向這個新new出來的對象。而changeObjValue方法修改的pt指向的對象的內容也就是修改的是堆內存的數據,ref和pt指向的是一塊兒內存。所以pt的對象數據會變化。

String和基本數據類型一樣傳的都是值

引用類型傳的是地址 好記一點,但是要明白內存關系。

String與StringBuffer的區別
簡單地說,就是一個變量和常量的關系。StringBuffer對象的內容可以修改;而String對象一旦產生後就不可以被修改,重新賦值其實是兩個對象。
StringBuffer的內部實現方式和String不同,StringBuffer在進行字符串處理時,不生成新的對象,在內存使用上要優於String類。所以在實際使用時,如果經常需要對一個字符串進行修改,例如插入、刪除等操作,使用StringBuffer要更加適合一些。
String:在String類中沒有用來改變已有字符串中的某個字符的方法,由於不能改變一個Java字符串中的某個單獨字符,所以在JDK文檔中稱String類的對象是不可改變的。然而,不可改變的字符串具有一個很大的優點:編譯器可以把字符串設為共享的。
StringBuffer:StringBuffer類屬於一種輔助類,可預先分配指定長度的內存塊建立一個字符串緩沖區。這樣使用StringBuffer類的append方法追加字符 比 String使用 + 操作符添加字符 到 一個已經存在的字符串後面有效率得多。因為使用 + 操作符每一次將字符添加到一個字符串中去時,字符串對象都需要尋找一個新的內存空間來容納更大的字符串,這無凝是一個非常消耗時間的操作。添加多個字符也就意味著要一次又一次的對字符串重新分配內存。使用StringBuffer類就避免了這個問題。
StringBuffer是線程安全的,在多線程程序中也可以很方便的進行使用,但是程序的執行效率相對來說就要稍微慢一些。
StringBuffer的常用方法
StringBuffer類中的方法要偏重於對字符串的變化例如追加、插入和刪除等,這個也是StringBuffer和String類的主要區別。
1、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”。
2、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”。
3、insert方法
public StringBuffer insert(int offset, String s)
該方法的作用是在StringBuffer對象中插入內容,然後形成新的字符串。例如:
StringBuffer sb = new StringBuffer(“TestString”);
sb.insert(4,“false”);
該示例代碼的作用是在對象sb的索引值4的位置插入字符串false,形成新的字符串,則執行以後對象sb的值是”TestfalseString”。
4、reverse方法
public StringBuffer reverse()
該方法的作用是將StringBuffer對象中的內容反轉,然後形成新的字符串。例如:
StringBuffer sb = new StringBuffer(“abc”);
sb.reverse();
經過反轉以後,對象sb中的內容將變為”cba”。
5、setCharAt方法
public void setCharAt(int index, char ch)
該方法的作用是修改對象中索引值為index位置的字符為新的字符ch。例如:
StringBuffer sb = new StringBuffer(“abc”);
sb.setCharAt(1,’D’);
則對象sb的值將變成”aDc”。
6、trimToSize方法
public void trimToSize()
該方法的作用是將StringBuffer對象的中存儲空間縮小到和字符串長度一樣的長度,減少空間的浪費。
7、構造方法:
StringBuffer s0=new StringBuffer();分配了長16字節的字符緩沖區
StringBuffer s1=new StringBuffer(512);分配了512字節的字符緩沖區
8、獲取字符串的長度: length()
StringBuffer s = new StringBuffer("www");
int i=s.length();
m.返回字符串的一部分值
substring(int start) //返回從start下標開始以後的字符串
substring(int start,int end) //返回從start到 end-1字符串
9.替換字符串
replace(int start,int end,String str)
s.replace(0,1,"qqq");
10.轉換為不變字符串:toString()。

String String Buffer String Builder