壹、從根本上認識java.lang.String類和String池
首先,我建議先看看String類的源碼實現,這是從本質上認識String類的根本出發點。從中可以看到:
1、String類是final的,不可被繼承。public final class String。
2、String類是的本質是字符數組char[], 並且其值不可改變。private final char value[];
然後打開String類的API文檔,可以發現:
3、String類對象有個特殊的創建的方式,就是直接指定比如String x = "abc","abc"就表示壹個字符串對象。而x是"abc"對象的地址,也叫
做"abc"對象的引用。
4、String對象可以通過“+”串聯。串聯後會生成新的字符串。也可以通過concat()來串聯,這個後面會講述。
6、Java運行時會維護壹個String Pool(String池),JavaDoc翻譯很模糊“字符串緩沖區”。String池用來存放運行時中產生的各種字符串,
並且池中的字符串的內容不重復。而壹般對象不存在這個緩沖池,並且創建的對象僅僅存在於方法的堆棧區。
5、創建字符串的方式很多,歸納起來有三類:
其壹,使用new關鍵字創建字符串,比如String s1 = new String("abc");
其二,直接指定。比如String s2 = "abc";
其三,使用串聯生成新的字符串。比如String s3 = "ab" + "c";
二、String對象的創建
String對象的創建也很講究,關鍵是要明白其原理。
原理1:當使用任何方式來創建壹個字符串對象s時,Java運行時(運行中JVM)會拿著這個X在String池中找是否存在內容相同的字符串對象,
如果不存在,則在池中創建壹個字符串s,否則,不在池中添加。
原理2:Java中,只要使用new關鍵字來創建對象,則壹定會(在堆區或棧區)創建壹個新的對象。
原理3:使用直接指定或者使用純字符串串聯來創建String對象,則僅僅會檢查維護String池中的字符串,池中沒有就在池中創建壹個,有則罷
了!但絕不會在堆棧區再去創建該String對象。
原理4:使用包含變量的表達式來創建String對象,則不僅會檢查維護String池,而且還會在堆棧區創建壹個String對象。
另外,String的intern()方法是壹個本地方法,定義為public native String intern(); intern()方法的價值在於讓開發者能將註意力集中到
String池上。當調用 intern 方法時,如果池已經包含壹個等於此 String 對象的字符串(該對象由 equals(Object) 方法確定),則返回池
中的字符串。否則,將此 String 對象添加到池中,並且返回此 String 對象的引用。
三、不可變類
不可改變的字符串具有壹個很大的優點:編譯器可以把字符串設置為***享。
不可變類String有壹個重要的優點-它們不會被***享引用。
是這樣的,JAVA為了提高效率,所以對於String類型進行了特別的處理---為string類型提供了串池
定義壹個string類型的變量有兩種方式:
string name= "tom ";
string name =new string( "tom ")
使用第壹種方式的時候,就使用了串池,
使用第二中方式的時候,就是壹種普通的聲明對象的方式
如果妳使用了第壹種方式,那麽當妳在聲明壹個內容也是 "tom "的string時,它將使用串池裏原來的那個內存,而不會重新分配內存,也就是說,string saname= "tom ",將會指向同壹塊內存
另外關於string類型是不可改變的問題:
string類型是不可改變的,也就是說,當妳想改變壹個string對象的時候,比如name= "madding "
那麽虛擬機不會改變原來的對象,而是生成壹個新的string對象,然後讓name去指向它,如果原來的那個 "tom "沒有任何對象去引用它,虛擬機的垃圾回收機制將接收它。
據說這樣可以提高效率!!!