Java基礎類學習之String詳解
1 String不可變性
- String類被聲明為 final,因此它不可被繼承。
- 內(nèi)部使用char數(shù)組存儲數(shù)據(jù),該數(shù)組被聲明為final,這意味著value數(shù)組初始化之后就不能再指向其它數(shù)組。
- String內(nèi)部沒有改變value數(shù)組的方法
- String類中所有修改String值的方法,如果內(nèi)容沒有改變,則返回原來的String對象引用,如果改變了,創(chuàng)建了一個全新的String對象,包含修改后的字符串內(nèi)容,最初的String對象沒有任何改變。(目的:節(jié)約存儲空間、避免額外的開銷)
//String的類聲明以及value字段代碼:
public final class String
implements java.io.Serializable, Comparable<String>, CharSequence {
private final char value[]; //字符數(shù)組存儲String的內(nèi)容
/** Cache the hash code for the string */
private int hash; // Default to 0
}
不可變的驗證分析:
public class Immutable {
public static String upcase(String s) {
return s.toUpperCase();
}
public static void main(String[] args) {
String q = "howdy";
System.out.println(q); // howdy
String qq = upcase(q);
System.out.println(qq); // HOWDY
System.out.println(q); // howdy
}
} /* 輸出:
howdy
HOWDY
howdy
*///:~
- 當把q傳給upcase0方法時,實際傳遞的是引用的一個拷貝。
- upcase0方法中,傳入引用s,只有upcase0運行的時候,局部引用s才存在。一旦upcase0運行結(jié)束,s就消失。upcaseO的返回值是最終結(jié)果的引用。
- 綜上,upcase()返回的引用已經(jīng)指向了一個新的對象,而原本的q則還在原地。
延伸結(jié)論:
String對象作為方法的參數(shù)時,都會復制一份引用,參數(shù)傳遞是引用的拷貝
2 不可變的好處
1. 可以緩存 hash 值
String的hash值經(jīng)常被使用,例如String用做HashMap的key。不可變的特性可以使得hash值也不可變,因此只需要進行一次計算。
2. String Pool 的需要
如果一個String對象已經(jīng)被創(chuàng)建過了,那么就會從 String Pool 中取得引用。只有String是不可變的,才可能使用 String Pool。
3. 線程安全
String不可變性天生具備線程安全,可以在多個線程中安全地使用。
3 String+和StringBuilder效率差異
String使用“+”表示字符串拼接
先說結(jié)論:
- “+”操作,javac編譯器會自動優(yōu)化為StringBuilder.append() 調(diào)用。
- StringBuilder要比“+”操作高效
- 涉及循環(huán)追加的,手動創(chuàng)建StringBuilder對象操作比“+”操作編譯器優(yōu)化,更高效
驗證:
public class StringBuilderTest {
public static void main(String[] args) {
String s1 = "ABC";
String s2 = "123";
String result = s1+s2;
System.out.println(result);
}
}
編譯并查看字節(jié)碼:javap -verbose StringBuilderTest.class

執(zhí)行過程:
調(diào)用了2次append()方法,最后調(diào)用StringBuilder.toString()返回最終結(jié)果
為什么StringBuilder要比+高效?
- +操作,按照:每次追加都創(chuàng)建新的String對象,把字符加入value數(shù)組中。這里產(chǎn)生一次對象創(chuàng)建操作,以及對應的垃圾回收
- StringBuilder的底層數(shù)組value也是用到了char[],但它沒有聲明為final,故它可變,所以追加內(nèi)容時不用創(chuàng)建新的數(shù)組,而是直接修改value
- StringBuilder比+省去String對象創(chuàng)建以及垃圾回收的開銷,因此效率更高。
源碼追溯:
//StringBuilder.append()
@Override
public StringBuilder append(char c) {
super.append(c);
return this;
}
// 父類 AbstractStringBuilder.append()
@Override
public AbstractStringBuilder append(char c) {
ensureCapacityInternal(count + 1);
value[count++] = c;
return this;
}
//AbstractStringBuilder value 字段
abstract class AbstractStringBuilder implements Appendable, CharSequence {
//The value is used for character storage.
char[] value; // 沒有聲明為final,因此value可變
}
手動實現(xiàn)StringBuilder對象操作比編譯器自行優(yōu)化,更高效:
- 通過字節(jié)碼分析可知(我這里省去了,可以自己實現(xiàn)驗證):循環(huán)部分的代碼更簡短、更簡單,而且它只生成了一個StringBuilder對象。
- 顯式地創(chuàng)建StringBuilder還允許你預先為其指定大小。如果你已經(jīng)知道最終的字符串大概有多長,那預先指定StringBuilder的大小可以避免多次重新分配緩沖。
當你為一個類編寫toString方法時,如果字符串操作比較簡單,那就可以信賴編譯器,它會為你合理地構(gòu)造最終的字符串結(jié)果。但是,如果你要在toString0方法中使用循環(huán),那么最好自己創(chuàng)建一個StringBuilder對象來實現(xiàn)。
4 String, StringBuffer and StringBuilder
可變性
- String 不可變
- StringBuffer和StringBuilder可變
線程安全
- String不可變,因此是線程安全的
- StringBuilder 不是線程安全的
- StringBuffer 是線程安全的,內(nèi)部使用synchronized進行同步
效率
- 如果要操作少量的數(shù)據(jù)用String
- 單線程環(huán)境且字符串緩沖區(qū)涉及大量數(shù)據(jù) StringBuilder
- 多線程環(huán)境且字符串緩沖區(qū)涉及大量數(shù)據(jù) StringBuffer
5 String與JVM內(nèi)存管理
一、引入字符串常量池
- Javac編譯后,字節(jié)碼文件中有一塊區(qū)域:常量池,存儲了包括類中聲明的字符串常量值等字面量
- 運行時,JVM開辟實際內(nèi)存空間:字符串常量值寫入了字符串常量池,屬于方法區(qū)的一部分。
案例1:
String s1 = "win"; String s2 = "win"; System.out.println(s1==s2); //輸出結(jié)果:true //引用 s1 s2 的值等于win在字符串常量池的地址值
結(jié)論:
引用 s1 s2 的值等于win在字符串常量池的地址值
分析字節(jié)碼的執(zhí)行過程:

案例2:
public class StringPool {
public static void main(String[] args) {
String s3 = new String("win");
String s4 = new String("win");
System.out.println(s3==s4);//false
}
}
結(jié)論:
通過new操作符創(chuàng)建的字符串對象不指向字符串池中的任何對象。
字節(jié)碼分析:

綜上:
public class StringPool {
public static void main(String[] args) {
String s1 = "win";
String s2 = "win";
String s3 = new String("win");
String s4 = new String("win");
System.out.println(s1==s2);//true
System.out.println(s1==s3);//false
System.out.println(s3==s4);//false
}
}
6 String api方法
從這個表中可以看出,當需要改變字符串的內(nèi)容時,String類的方法都會返回一個新的String對象。同時,如果內(nèi)容沒有發(fā)生改變,String的方法只是返回指向原對象的引用而已。這可以節(jié)約存儲空間以及避免額外的開銷。


以上就是Java基礎類學習之String詳解的詳細內(nèi)容,更多關(guān)于Java String類的資料請關(guān)注腳本之家其它相關(guān)文章
相關(guān)文章
Spring?Boot使用MyBatis進行兩個表的關(guān)聯(lián)
本文主要介紹了Spring?Boot使用MyBatis進行兩個表的關(guān)聯(lián),通過實例演示了如何使用MyBatis的XML映射文件和注解實現(xiàn)關(guān)聯(lián)操作,具有一定的參考價值,感興趣的可以了解一下2023-09-09
Spring Boot RestTemplate提交表單數(shù)據(jù)的三種方法
本篇文章主要介紹了Spring Boot RestTemplate提交表單數(shù)據(jù)的三種方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-03-03
JAVA系統(tǒng)中Spring?Boot應用程序的配置文件application.yml使用詳解
這篇文章主要介紹了JAVA系統(tǒng)中Spring?Boot應用程序的配置文件application.yml的相關(guān)資料,application.yml是Spring?Boot應用程序的配置文件,定義了服務器、Spring、日志、安全及其他配置屬性,確保應用程序正確啟動和運行,需要的朋友可以參考下2025-01-01
Intellij Idea插件開發(fā)之創(chuàng)建項目層級的右鍵菜單
這篇文章主要介紹了Intellij Idea插件開發(fā)之創(chuàng)建項目層級的右鍵菜單,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-02-02
java中hashCode方法與equals方法的用法總結(jié)
總的來說,Java中的集合(Collection)有兩類,一類是List,再有一類是Set。前者集合內(nèi)的元素是有序的,元素可以重復;后者元素無序,但元素不可重復2013-10-10
Java 實現(xiàn)簡易教務管理系統(tǒng)的代碼
這篇文章主要介紹了Java 實現(xiàn)簡易教務管理系統(tǒng)的代碼,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-07-07

