1. 程式人生 > >Java中常量池詳解

Java中常量池詳解

在Java的記憶體分配中,總共3種常量池: 轉發連結 :https://blog.csdn.net/zm13007310400/article/details/77534349

1.字串常量池(String Constant Pool):

1.1:字串常量池在Java記憶體區域的哪個位置?

  • 在JDK6.0及之前版本,字串常量池是放在Perm Gen區(也就是方法區)中;
  • 在JDK7.0版本,字串常量池被移到了堆中了。至於為什麼移到堆內,大概是由於方法區的記憶體空間太小了。
  • JDK8以後也還是放在了Heap空間中,並沒有已到元空間。

1.2:字串常量池是什麼?

  • 在HotSpot VM裡實現的string pool功能的是一個StringTable類,它是一個Hash表,預設值大小長度是1009;這個StringTable在每個HotSpot VM的例項只有一份,被所有的類共享。字串常量由一個一個字元組成,放在了StringTable上。
  • 在JDK6.0中,StringTable的長度是固定的,長度就是1009,因此如果放入String Pool中的String非常多,就會造成hash衝突,導致連結串列過長,當呼叫String#intern()時會需要到連結串列上一個一個找,從而導致效能大幅度下降;
  • 在JDK7.0中,StringTable的長度可以通過引數指定:
  1.  -XX:StringTableSize=66666

1.3:字串常量池裡放的是什麼?

  • 在JDK6.0及之前版本中,String Pool裡放的都是字串常量;
  • 在JDK7.0中,由於String#intern()發生了改變,因此String Pool中也可以存放放於堆內的字串物件的引用。關於String在記憶體中的儲存和String#intern()方法的說明,可以參考我的另外一篇部落格:

 

需要說明的是:字串常量池中的字串只存在一份!    如:   1. String s1 = "hello,world!";   2. String s2 = "hello,world!";   即執行完第一行程式碼後,常量池中已存在 “hello,world!”,那麼 s2不會在常量池中申請新的空間,而是直接把已存在的字串記憶體地址返回給s2。(這裡具體的字串如何分配就不細說了,可以看我的另一篇部落格)

2.class常量池(Class Constant Pool):

2.1:class常量池簡介:

  • 我們寫的每一個Java類被編譯後,就會形成一份class檔案;class檔案中除了包含類的版本、欄位、方法、介面等描述資訊外,還有一項資訊就是常量池(constant pool table),用於存放編譯器生成的各種字面量(Literal)和符號引用(Symbolic References);
  • 每個class檔案都有一個class常量池。

2.2:什麼是字面量和符號引用:

  • 字面量包括:1.文字字串 2.八種基本型別的值 3.被宣告為final的常量等;
  • 符號引用包括:1.類和方法的全限定名 2.欄位的名稱和描述符 3.方法的名稱和描述符。

3.執行時常量池(Runtime Constant Pool):

  • 執行時常量池存在於記憶體中,也就是class常量池被載入到記憶體之後的版本,不同之處是:它的字面量可以動態的新增(String#intern()),符號引用可以被解析為直接引用
  • JVM在執行某個類的時候,必須經過載入、連線、初始化,而連線又包括驗證、準備、解析三個階段。而當類載入到記憶體中後,jvm就會將class常量池中的內容存放到執行時常量池中,由此可知,執行時常量池也是每個類都有一個。在解析階段,會把符號引用替換為直接引用,解析的過程會去查詢字串常量池,也就是我們上面所說的StringTable,以保證執行時常量池所引用的字串與字串常量池中是一致的。
  
public static void main(String[] args) {
        String s1="abc";
        String s2="abc";
        System.out.println(s1==s2);//true
        String s3=new String("ab")+new String("c");
         
        // 這裡顯示false,可能先在字串常量池中新增s1,然後動態的新增s3,會顯示false
        s3.intern();    
        System.out.println(s1==s3); //false
          
        String s4=new String("ab")+new String("cd");
        //s4.intern();
        String s5="abcd";
        System.out.println(s4==s5);//true,去掉intern之後是false;
          
    }