JVM之1.7于1.8内存区别
jvm的内存模型在1.7和1.8有较大区别。
1、jdk1.7的堆内存
-
Young 年轻代
Young区被划分为三个部分,Eden区和两个大小严格相同的Survivor区,其中,Survivor区间中,某一时刻只有其中一个是被使用的,另一个留做垃圾收集时复制对象用,在Eden区间变满的时候,GC就会将存活得的对象移到空闲的Survivor区间中,根据JVM的策略,在经过几次垃圾收集后,任然存活于Survivor的对象将被移到Tenured区间。
-
Tenured 老年代
Tenured区主要保存生命周期长的对象,一般是一些老的对象,当一些对象在Young复制转移一定次数以后,对象就会被转移到Tenured区,一般如果系统中用了Application级别的缓存,缓存中的对象往往会被转移到这一区间。
-
Perm 永久代
Perm代主要保存class,method,filed对象,这部分的空间一般不会溢出,除非一次性加载了很多的类,不过在涉及热部署的应用服务器的时候,有时候会遇到java.lang.OutOfMemoryError:PermGen space的错误,造成这个错误的很大原因就有可能是每次都重新部署,但是重新部署后,类的class没有被卸载掉,这就造成了大量的class对象保存在Perm中,这种情况下,一般重新启动应用服务器可以解决问题。
-
Virtual
- 最大内存和初始内存的差值,就是Virtual区。
2、jdk1.8的堆内存
由上图可以看出,jdk1.8的内存模型是由两部分组成,年轻到 + 老年代。
年轻代:Eden + 2 * Survivor。
老年代:OldGen。
在jdk1.8中变化最大的就是Perm区,用Metaspace(元数据空间)进行替换。
注意:Metaspace所占用的内存空间不是在虚拟机内部,而是在本地内存空间中,这也是与1.7的永久代最大的区别所在。
现实使用中,由于永久代内存经常不够用或者发生内存泄漏,爆出异常:java.lang.OutOfMemoryError:PermGen。基于此,将永久代废弃,而改用元空间,改为使用本地内存空间。