1. 程式人生 > >java 創建string對象機制 字符串緩沖池 字符串拼接機制 字符串中intern()方法

java 創建string對象機制 字符串緩沖池 字符串拼接機制 字符串中intern()方法

結果 減少 存在 pub lse ring 方法 機制 引用

字符串常量池:字符串常量池在方法區中

為了優化空間,為了減少在JVM中創建的字符串的數量,字符串類維護了一個字符串池,每當代碼創建字符串常量時,JVM會首先檢查字符串常量池。如果字符串已經存在池中,就返回池中的實例引用。如果字符串不在池中,就會實例化一個字符串並放到池中。Java能夠進行這樣的優化是因為字符串是不可變的,可以不用擔心數據沖突進行共享。所以,在常量池中的這些字符串不會被垃圾收集器回收

1.String str = new String("hello");此時創建的2個對象,但是引用str只指向堆中的對象,並沒有指向常量池中的對象

創建了2個對象,1.檢查常量池中有沒有hello,沒有的話,創建對象放到常量池中,再創建對象放到堆中。如果常量池有hello對象,則只創建一個對象並放到堆中。

2.字符串常量池在方法區

3.String str = "hello";此時創建1個對象,引用str指向常量池中的對象

檢查常量池有無hello,如果有,則把指向該對象,如果沒有,創建對象放在常量池裏。

4.intern()方法。把字符串變成常量池裏的字符串,即是:把引用指向常量池中的對象

如果常量池中已經包含了等於該String對象的字符串,則返回該字符串。否則,將該String對象加入常量池,並返回引用。

5.String s1 = "tom";

 String s2 = "cat";

 String s3 = "tomcat";//創建了一個對象,常量池中。

 String s4 = "tom"+"cat";//不創建對象,jvm認為帶雙引號的字符串拼接就是一個整體,並且常量池中已經存在tomcat,故此處不創建對象

String s5 = s1 + s2;//創建一個對象"tomcat"

s3 == s4;true

 s3 == s5;false

為什麽為false呢?

字符串引用的拼接並不等於帶雙引號的字符串的拼接,原來兩個字符串s1, s2的拼接首先會調用 String.valueOf(obj),這個Obj為s1,而String.valueOf(Obj)中的實現是return obj == null ? "null" : obj.toString(), 然後產生StringBuilder, 調用的StringBuilder(s1)構造方法, 把StringBuilder初始化,長度為str1.length()+16。此時的StringBuilder對象是在堆上創建的!, 接下來調用StringBuilder.append(str2), 把第二個字符串拼接進去, 然後調用StringBuilder.toString返回結果。所以會返回false。

調用了String.valueOf(),產生了stringBuilder,調用它的構造方法,在堆上new出了新對象,然後使用append()方法把第二個字符串s2拼接起來。

實例:

public static void main(String[] args) {
    String str1 = new String("abc");
    String str2 = "abc";
    String str3 = new String("abc");
    System.out.println(str1 == str2.intern());
    System.out.println(str1 == str3.intern());
    System.out.println(str2 == str3.intern());
    System.out.println(str2 == str1.intern());
}
技術分享圖片

  結果:false false true true

  首先intern()方法:

  public String intern()返回字符串對象的規範化表示形式。(這句話到底啥意思我也不太清楚)
  當調用 intern 方法時,如果池已經包含一個等於此 String 對象的字符串(該對象由 equals(Object) 方法確定),則返回池中的字符串。否則,將此 String 對象添加到池中,並且返回此 String 對象的引用。 它遵循對於任何兩個字符串 s 和 t,當且僅當 s.equals(t) 為 true 時,s.intern() == t.intern() 才為 true。

  也就是說intern()方法返回的字符串對象肯定是池中的對象而且字符串內容和調用該方法的對象的內容一樣。那麽結果是false false true true也就不難理解了。str3.intern()和str1.intern()返回的對象就是str2所指向的對象呀。所以我們的結論也得以驗證。

java 創建string對象機制 字符串緩沖池 字符串拼接機制 字符串中intern()方法