Java 虚拟机(JVM)主要分为以下几个区域:程序计数器、虚拟机栈、本地方法栈、堆、方法区/元空间。其中除了程序计数器和本地方法栈比较小而且不容易出现内存溢出,其它区域都可能发生内存溢出的情况。
以下是各个区域可能出现内存溢出的场景:
1. 程序计数器
程序计数器是用于记录当前线程正在执行的 JVM 字节码指令的地址。它的空间非常小,因此几乎不会发生内存溢出的情况。
2. 虚拟机栈
虚拟机栈是用于存储局部变量表、操作数栈、动态链接、方法出口等信息的区域。每个线程在执行方法时都会创建一个栈帧,存储对应的方法信息。如果栈空间不足,就会抛出 StackOverflowError 异常;如果线程请求分配的栈容量超过了 JVM 允许的最大容量,就会抛出 OutOfMemoryError 异常。
3. 本地方法栈
本地方法栈与虚拟机栈的作用类似,不同的是本地方法栈是为 JVM 调用 Native 方法服务的。与虚拟机栈一样,本地方法栈也有 StackOverflowError 和 OutOfMemoryError 两种内存溢出异常。
4. 堆
堆是用于存储 Java 对象实例的区域,是 JVM 中内存最大的一块。如果创建太多对象、无法释放已经不再使用的对象,堆空间就会被耗尽,此时就会抛出 OutOfMemoryError 异常。一些常见的导致堆内存溢出的场景包括:循环创建大量对象、持有大量对象的集合没有及时清理、静态集合类的值长时间未被使用等。
5. 方法区/元空间
方法区/元空间用于存储类信息、常量、静态变量、即时编译器编译后的代码等。如果加载的类过多或者常量池中保存的常量过多、动态代理导致反复生成的类型过多,都有可能导致方法区/元空间的内存溢出。在 JDK7 及以前版本,该区域通常被称为 “永久代”,而在 JDK8 以后的版本,该区域被移到了本地内存中,称为 “元空间”。遇到方法区/元空间内存溢出,通常会抛出 OutOfMemoryError 异常。