首先我們來看看JDK中的String類別源碼,它實作了很多接口,可以看到String類別被final修飾了,這表示String類別不可以被繼承,String不存在子類,這樣所有使用JDK的人,用到的String類別都是同一個,如果String允許被繼承,每個人都可以對String進行擴展,每個人使用的String都不是同一個版本,兩個不同的人使用相同的方法,表現出不同的結果,這就導致程式碼沒辦法進行開發了
繼承和方法覆寫在帶來彈性的同時,也會帶來許多子類別行為不一致的問題
方式一:直接賦值(常用)
String str = " hello word "
方式二:透過建構方法產生物件
String str1 = new String(" hello word ");
方式三:透過字元陣列產生物件
char[] data = new char[]{'a' , 'b' ,'c'};
方式四:透過String的靜態方法valueOf(任意資料型態) = >轉為字串(常用)
String str2 = String.valueOf(10);
字面量:直接寫出來的數值叫做字面量
10 – > int字面量
10.1 --> double字面量
true --> boolean字面量
" abc " – > String字面量
字串的字面量其實就是一個字串物件
String str = “hello word”;
String str2 = str;
此時這既是一個字串的字面量,也是字串的對象,為了方便理解,畫個圖,此時是為了方便理解我們暫時先認為它儲存在堆上,其實在方法區中存放
# 若此時讓str2 = “Hello”;此時對str的輸出並沒有影響,因為被" "括起來的Hello也是一個字符串對象,說明此時在堆上新開闢了一塊空間,而此時str2保存的就是新物件的位址空間,對str沒有影響
所有的引用資料型別在比較是否相等時,使用equals方法比較,JDK的常用類,都已經覆寫了equals方法,直接使用即可
引用數據類型使用== 比較的是地址
下圖是兩個引用指向了同一塊地址空間,和字串的常數池有關
下圖產生了兩個對象,兩塊位址空間,使用==返回的就是false
equals的比較大小是區分大小寫的比較
equalsIgnoreCase方法是不區分大小寫的比較
# 當使用直接賦值法產生字串物件的時候,JVM會維護一個字串的常數池,若該物件在堆中還不存,就產生一個字串物件加入到字串常數池中;當繼續使用直接賦值法產生字串物件的時候,JVM發現該引用指向的內容在常數池中已經存在了,此時就不再新建字符串對象,而是直接復用已有的對象,這也是為什麼上圖的三個引用指向的是同一塊地址
當第一次產生物件的時候,常數池中還什麼都沒有,就在常數池中產生一個字串物件存入,當第二第三次產生物件時,JVM發現常數池中已經存在相同的內容,就不再產生新的對象,直接指向和str1相同的位址空間
程式都是從右向左執行的,此時第一行程式碼的右邊就是一個字串常數,也是一個字串對象,所以先在常數池中開闢一塊空間,然後新建一個字串物件存入,程式再往左執行,遇到new關鍵字,此時新建一個物件存入堆中,然後str1 指向堆中的對象,在指向第二行第三行程式碼時,發現常數池中已經存在該對象,不再新建,遇到new關鍵字就新建對象,內存圖如下:
呼叫intern方法會將目前字串引用指向的物件保存到字串常數池中,有兩種情況:
1.若當前常數池中已經存在了該對象,則不再產生新的對象,返回常量池中的String對象
2.若當前常數池中不存在該對象,則將此物件入池,返回入池後的位址。
因為intern方法是有回傳值的,此時str1只是呼叫了intern方法,並沒有接收回傳值,所以str1還是指向堆中的對象,str2指向常數池中的對象,所以回傳false;
只要接收一下呼叫intern方法的回傳值,就會回傳true;
此時就將str1指向的物件手動入池了,池中已有該對象,直接讓str1指向該物件
2.再看看下面這幾行程式碼的輸出
手動入池時,池中還沒有任何東西,直接移入常數池中
這裡的不可變指的是" hello " , " world " , " helloworld " , " !!! " , 以及拼接後的"helloworld!!!"這些已經創建好的字符串對象,這些對像一旦聲明後來就無法修改其內容,但引用是可以改變的,一會指向hello,一會指向helloworld,一會指向hello world! ! ! ,這都是可以的
字串就是就是一個字元陣列—> char[],字串其實在字元陣列中保存。字串的內容為什麼不能改變?我們看看字串的源碼就知道了。
我們可以看到String內部的字元陣列是被封裝起來的,String類別的外部無法存取到這個字元數組,更何談改變字串的內容
String str = " hello ";
2.更換使用StringBuilder或StringBuffer類別- - 已經不是一個類型了
a.StringBuilder:線程不安全,性能較強
b.StringBuffer:線程安全,性能較差
除此之外兩個類別的用法完全相同
StringBuilder類別和String類別的相互轉換:
其他常用方法:
a.字串的反轉操作,sb提供的reverse();
b.刪除指定範圍的數據,delete(int start,int end);刪除從start開始,到end之前的所有內容,左閉右開區間
c .插入操作,insert(int start,各種資料型別):從start索引位置開始插入,插入的起始索引為start
以上是java的String類別如何使用的詳細內容。更多資訊請關注PHP中文網其他相關文章!