jdk8前后内存分布的变化
内存分布变化主要集中在方法区上
jdk1.6时方法区的实现是在永久代上,使用jvm的内存,方法区中包括类信息(修饰符,全类名),域信息,方法信息,静态变量(引用在永久代,变量引用对象在堆上),运行时变量池,JIT代码缓存。
jdk1.7时将静态变量和StringTable(字符串常量池)移到了堆中,还保留了永久代
jdk1.8开始移除了永久代,方法区的实现改为了元空间,使用本地内存,静态变量(引用和变量对象实体都在堆空间)和StringTable放在对空间中。
为什么要移除永久代
1.永久代大小难以确定
永久代使用jvm内存,需要用户管理分配空间,但空间大小设置难以确定,在某些场景下,如果动态加载过多的类就会造成full gc(甚至OOM)造成STW,阻塞用户线程。而元空间使用本地内存,因此默认情况下,元空间大小受本地内存限制。
2.对永久代调优比较困难
为什么把StringTable放到堆中
因为永久代的回收效率很低,在full gc的时候才会触发回收。这导致了StringTable回收率不高,而我们开发中会有大量的字符串被创建,回收率低导致永久代内存不足。而放到堆中能够及时的被回收。