方法区的演变细节
(参考:https://openjdk.java.net/jeps/122)
- 首先明确:只有Hotspot中才有永久代。BEA、JRockit、IBM的J9等来说,是不存在永久代的概念的。原则上如何实现方法区属于虚拟机实现细节,不受《Java虚拟机规范》约束,不做统一要求
- Hotspot中方法区的变化
- jdk1.6及以前:有永久代,静态变量存放在永久代上
- jdk1.7:有永久代,但是已经逐步“去永久代”,字符串常量池、静态变量移除,保存在堆中
- jdk1.8及之后:无永久代,类型信息、字段、方法、常量保存在本地内存的元空间,但是字符串常量池、静态变量仍在堆中
永久代为什么要被元空间替换?
- 随着Java8的到来,Hotspot VM再也看不到永久代。但是这并不意味着类的元数据信息消失了。这些数据被移到了与堆不相连的本地内存区域,这个区域叫做元空间(Matespace)
- 由于类的元空间分配到本地内存中,元空间的最大可分配空间就是系统可用的内存空间
- 这项改动是很有必要的:
- 为永久代设置空间大小是很难确定的
- 对永久代进行调优是很困难
StringTable为什么要调整?
jdk7中将StringTable放到了堆空间。因为永久代的回收效率很低,在Full GC的时候才会触发。而在Full GC是老年代的空间不足、永久代空间不足的时候才会触发。这就导致StringTable回收效率不高。而我们开发中会有大量的字符串被创建,回收效率低,导致永久代内存空间不足。放到堆里,能及时回收内存。