1.常量池的概念
分為兩大類(lèi):靜態(tài)常量池和運(yùn)行時(shí)常量池。
- class文件靜態(tài)常量池
即class文件中的常量池, 指的是編譯生成的 class 字節(jié)碼文件,其結(jié)構(gòu)中有一項(xiàng)是常量池(Constant Pool Table)
,用于存放編譯期生成的各種字面量和符號(hào)引用,這部分內(nèi)容將在類(lèi)加載后進(jìn)入內(nèi)存的運(yùn)行時(shí)常量池中存放
。該常量池不僅僅包含字符串(數(shù)字)字面量
,還包含類(lèi)、方法的信息,占用class文件絕大部分空間。字面量包括
:(1)文本字符串 (2)八種基本類(lèi)型的值 (3)被聲明為final的常量等; - 運(yùn)行時(shí)常量池
是方法區(qū)中的一塊內(nèi)存區(qū)域,jvm虛擬機(jī)在完成類(lèi)裝載操作后,將class文件中的常量池載入到內(nèi)存中,我們常說(shuō)的常量池,就是運(yùn)行時(shí)常量池。一個(gè)類(lèi)加載到 JVM 中后對(duì)應(yīng)一個(gè)運(yùn)行時(shí)常量池
。Class 文件常量只是一個(gè)靜態(tài)存儲(chǔ)結(jié)構(gòu),里面的引用都是符號(hào)引用。而運(yùn)行時(shí)常量池可以在運(yùn)行期間將符號(hào)引用解析為直接引用。
注意: 運(yùn)行時(shí)常量池中的常量,基本來(lái)源于各個(gè)class文件中的常量池
。程序運(yùn)行時(shí),除非手動(dòng)向常量池中添加常量(比如調(diào)用intern方法),否則jvm不會(huì)自動(dòng)添加常量到常量池。
2.關(guān)于方法區(qū)的理解
-
方法區(qū)只是一個(gè)概念
:《Java虛擬機(jī)規(guī)范》只是規(guī)定了有方法區(qū)這么個(gè)概念和它的作用
,并沒(méi)有規(guī)定如何去實(shí)現(xiàn)它
。那么,在不同的 JVM 上方法區(qū)的實(shí)現(xiàn)肯定是不同的了
。 同時(shí)大多數(shù)用的JVM都是Sun公司的HotSpot。 - 方法區(qū)是Java虛擬機(jī)規(guī)范中的定義,是一種規(guī)范,在HotSpot的JMV中不同的jdk版本,
它的實(shí)現(xiàn)都不一樣。
(1)JDK7之前,HotSpot 使用永久代實(shí)現(xiàn)方法區(qū), 使用 GC分代來(lái)實(shí)現(xiàn)方法區(qū)內(nèi)存回收,可能出現(xiàn)“PermGen Space”的內(nèi)存溢出。
(2)JDK7,HotSpot 使用Java Heap或者Native memory實(shí)現(xiàn)方法區(qū)。可能會(huì)出現(xiàn)堆內(nèi)存溢出。
(3)JDK8,HotSpot 使用元空間(Metaspace)來(lái)實(shí)現(xiàn)。元空間并不在虛擬機(jī)中,而是使用本地內(nèi)存。因此,默認(rèn)情況下,元空間的大小僅受本地內(nèi)存限制。可以通過(guò)參數(shù)來(lái)指定元空間的大小。
補(bǔ)充兩個(gè)概念:
(1)方法區(qū)——線程共享的,主要存儲(chǔ)類(lèi)信息、常量池、靜態(tài)變量
、JIT編譯后的代碼等數(shù)據(jù)。方法區(qū)理論上來(lái)說(shuō)是堆的邏輯組成部分;
(2)運(yùn)行時(shí)常量池——是方法區(qū)的一部分,用于存放編譯期生成的各種字面量和符號(hào)引用;
3.字符串常量池中存放的時(shí)引用還是對(duì)象的問(wèn)題
- 字符串常量池存放的是對(duì)象引用,不是對(duì)象。在Java中,即使字符串是不可變的,它仍然和Java中的其他對(duì)象一樣。
對(duì)象都是創(chuàng)建在堆中,字符串也不例外
。所以字符串常量池仍然依靠堆,他們存儲(chǔ)的只是堆中字符串的引用。
- GC回收條件:當(dāng)一個(gè)對(duì)象
不再有引用指向它
時(shí),這個(gè)對(duì)象就會(huì)被回收。字符串字面量總是有一個(gè)來(lái)自字符串常量池的引用。因此字符串字面量不會(huì)被垃圾回收。絕對(duì)不會(huì)。
4.Java編譯過(guò)程(字符串字面量的處理)
(1)當(dāng)一個(gè).java文件被編譯成.class文件時(shí),和所有其他常量一樣,每個(gè)字符串字面量都通過(guò)一種特殊的方式被記錄下來(lái)。
(2)當(dāng)一個(gè).class文件被加載時(shí)(注意加載發(fā)生在初始化之前),JVM在.class文件中尋找字符串字面量。
(3)當(dāng)找到一個(gè)時(shí),JVM會(huì)檢查是否有相等的字符串在常量池(存放的地址引用)中存放了堆中引用
。
(4)如果找不到,就會(huì)在堆中創(chuàng)建一個(gè)對(duì)象,然后將它的引用存放在池中的一個(gè)常量表中。
(5)一旦一個(gè)字符串對(duì)象的引用在常量池中被創(chuàng)建,這個(gè)字符串在程序中的所有字面量引用都會(huì)被常量池中已經(jīng)存在的那個(gè)引用代替。