絕大部分 Java 程序員應(yīng)該都見過 "java.lang.OutOfMemoryError: PermGen space "這個異常拿愧。這里的 “PermGen space”其實指的就是方法區(qū)。不過方法區(qū)和“PermGen space”又有著本質(zhì)的區(qū)別氢架。前者是 JVM 的規(guī)范傻咖,而后者則是 JVM 規(guī)范的一種實現(xiàn),并且只有 HotSpot 才有 “PermGen space”岖研,而對于其他類型的虛擬機(jī)卿操,如 JRockit(Oracle)、J9(IBM) 并沒有“PermGen space”。由于方法區(qū)主要存儲類的相關(guān)信息害淤,所以對于動態(tài)生成類的情況比較容易出現(xiàn)永久代的內(nèi)存溢出扇雕。
移除永久代的工作從JDK1.7就開始了。JDK1.7中窥摄,存儲在永久代的部分?jǐn)?shù)據(jù)就已經(jīng)轉(zhuǎn)移到了Java Heap或者是 Native Heap镶奉。但永久代仍存在于JDK1.7中,并沒完全移除溪王,譬如符號引用(Symbols)轉(zhuǎn)移到了native heap腮鞍;字面量(interned strings)轉(zhuǎn)移到了java heap值骇;類的靜態(tài)變量(class statics)轉(zhuǎn)移到了java heap莹菱。
JDK 1.6下,會出現(xiàn)“PermGen Space”的內(nèi)存溢出吱瘩,而在 JDK 1.7和 JDK 1.8 中道伟,會出現(xiàn)堆內(nèi)存溢出,并且 JDK 1.8中 PermSize 和 MaxPermGen 已經(jīng)無效使碾。因此蜜徽,可以大致驗證 JDK 1.7 和 1.8 將字符串常量由永久代轉(zhuǎn)移到堆中,并且 JDK 1.8 中已經(jīng)不存在永久代的結(jié)論票摇。
元空間的本質(zhì)和永久代類似拘鞋,都是對JVM規(guī)范中方法區(qū)的實現(xiàn)。不過元空間與永久代之間最大的區(qū)別在于:元空間并不在虛擬機(jī)中矢门,而是使用本地內(nèi)存盆色。因此,默認(rèn)情況下祟剔,元空間的大小僅受本地內(nèi)存限制隔躲,但可以通過以下參數(shù)來指定元空間的大小:
-XX:MetaspaceSize物延,初始空間大小宣旱,達(dá)到該值就會觸發(fā)垃圾收集進(jìn)行類型卸載,同時GC會對該值進(jìn)行調(diào)整:如果釋放了大量的空間叛薯,就適當(dāng)降低該值浑吟;如果釋放了很少的空間,那么在不超過MaxMetaspaceSize時耗溜,適當(dāng)提高該值买置。
-XX:MaxMetaspaceSize,最大空間强霎,默認(rèn)是沒有限制的忿项。