简要回答
方法区实在虚拟机规范里面被定义的,不同的虚拟机对这个定义的实现不同,在HotSpot 虚拟机中在 jdk1.7
版本之前的方法区实现叫永久代(PermGen space
),jdk1.8
之后叫做元空间(Metaspace
)。
如 JRockit(Oracle)
、J9(IBM)
虚拟机有方法区 ,但是就没有永久代 PermGen space
所以我们以下讨论的都是基于 HotSpot
虚拟机。
详细回答
再谈运行时内存
从小刀前面写的文章中可以知道,方法区是运行时区域的一部分,且是各个线程共享内存的区域,它用于存储类的信息、常量、静态变量、即使时编译后的代码等数据。我们来看看这虚拟机内存的分布图:
我们再来看看方法区这块内存的存储数据,如下:
《Java虚拟机规范》中明确说明:“尽管所有的方法区在逻辑上是属于堆的一部分,但一些简单的实现可能不会选择去进行垃圾收集或者进行压缩。”但对于 HotSpotJVM 而言,方法区还有一个别名叫做Non-Heap(非堆),目的就是要和堆分开。
用大白话说就是,制定虚拟机规范的这批人觉得方法区因该是属于堆的,但是如果实现者嫌麻烦的话,在你自己的虚拟机实现里可以不把这部分当成堆
永久代(PermGen)
在jdk7
及以前,HotSpot
对方法区的实现是叫做永久代 Permanent Generation space
。
这里有个误区,很多人把 永久代等价于方法区,会说:“永久代就是方法区”,这样子表述不够精准。这里举个例子,在自然界中我们通过形态特征、栖息环境、生活习惯、分布范围、繁殖方式等特性来定义动物的类别。比如犬科动物的特征是“体形矫健,四肢细长,擅长跑,颜面部长,吻端突出,尾较粗长,尾毛一般浓密蓬松。。。”,但是我们不能说犬科是哈士奇或者犬科是中华田园犬。我们只能说哈士奇是犬科的一个品种。
所以说在JDK1.7
以前的HotSpot
虚拟机和jdk8
的虚拟机运行时内存的变化如下图:
我们平时应该都见过 java.lang.OutOfMemoryError: PermGen space
这个异常。这个就是永久代内存空间不能瞒住内存分配要求,虚拟机抛出的一个异常。
在jdk7
以前可以通过如下两个参数调整永久代空间
- 通过
-xx:Permsize
来设置永久代初始分配空间。默认值是20.75M
。 - 通过
-XX:MaxPermsize
来设定永久代最大可分配空间。32位机器默认是64M
,64位机器模式是82M
。
public class OomMock {
public static void main<