JVM 内存结构之堆的不同 JDK 版本
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 区。
jdk1.8 的堆内存模型
由上图可以看出,jdk1.8 的内存模型是由2部分组成,年轻代+年老代。
- 年轻代:
Eden+2*Survivor
- 年老代:
OldGen
jdk1.8 中变化最大的 Perm 区,用 Metaspace(元数据空间)进行了替换。Metaspace 所占用的内存空间不是在虚拟机内部,而是在本地内存空间中,这也是与 1.7 的永久代最大的区别所在。
为什么要废弃 1.7 中的永久区
官网给出了解释:http://openjdk.java.net/jeps/122
1 |
|
现实使用中,由于永久代内存经常不够用或发生内存泄露,爆出异常 java.lang.OutOfMemoryError: PermGen
。基于此,将永久区废弃,而改用元空间,改为了使用本地内存空间。