jvm 中 方法区 永久代 常量池 元数据 这个名字的含义以及关系

方法区:

是java虚拟机规范中定义的名字 各个虚拟机实现上有所不同
HostSpot虚拟机中
1.在jdk1.7 以及前的版本实现的方法区称为- - -永久代
2.在java 虚拟机的堆内存中分配
3.里面主要存放的内容:已经被虚拟机加载的类信息,常量,静态变量,即时编译后的代码等
4.内存回收:主要是常量池的回收 和类型的卸载- -目前的回收效果不好

运行时常量池:

分配在方法区(永久代)中的,**但是1.7版本把 字符串常量池 单独拿到了堆空间中 **
存的内容:用于存放编译期生成的各种字面量以及符号引用,时机可能是静态编译期 也可以是动态编译时候

元数据区

1.8版本,移除永久代,改为元数据区,元数据区分配在本地内存就是系统可用内存空间,字符串常量池 还是在堆内的
优点:元空间的最大可分配空间就是系统可用内存空间。不容易发生内存溢出的问题

永久代转元数据区的原因:

1.字符串存在永久代中,容易发生内存溢出的问题
2.类及犯法的信息比较难确定,因此对于永久代的大小指定比较困难,太小容易出现永久代溢出,太大则容易导致老年代溢出。
3.永久代会为GC带来不必要的复杂度

元数据 区存在的问题:内存碎片

元空间虚拟机采用了组块分配的形式,同时区块的大小由类加载器类型决定。类信息并不是固定大小,因此有可能分配的空闲区块和类需要的区块大小不同,这种情况下可能导致碎片存在。元空间虚拟机目前并不支持压缩操作,所以碎片化是目前最大的问题。

补录

在 Jdk6 以及以前的版本中,字符串的常量池是放在堆的Perm区的,Perm区是一个类静态的区域,主要存储一些加载类的信息,常量池,方法片段等内容,默认大小只有4m,一旦常量池中大量使用 intern 是会直接产生java.lang.OutOfMemoryError:PermGen space错误的。
在 jdk7 的版本中,字符串常量池已经从Perm区移到正常的Java Heap区域了。为什么要移动,Perm 区域太小是一个主要原因,
Jdk8已经直接取消了Perm区域,而新建立了一个元区域。应该是jdk开发者认为Perm区域已经不适合现在 JAVA 的发展了。

你可能感兴趣的:(jvm)