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
2
3
4
5
This is part of the JRockit and Hotspot convergence effort. JRockit
customers do not need to configure the permanent generation (since JRockit does not have a
permanent generation) and are accustomed to not configuring the permanent generation.

这是 JRockit 和 Hotspot 融合工作的一部分。JRockit 客户不需要配置永久代(因为JRockit 没有永久代),习惯不配置永久代。

现实使用中,由于永久代内存经常不够用或发生内存泄露,爆出异常 java.lang.OutOfMemoryError: PermGen。基于此,将永久区废弃,而改用元空间,改为了使用本地内存空间。


JVM 内存结构之堆的不同 JDK 版本
https://flepeng.github.io/021-Java-42-JVM-JVM-内存结构之堆的不同-JDK-版本/
作者
Lepeng
发布于
2024年4月19日
许可协议