有四个区域可能发生内存溢出异常
虚拟机栈、本地方法栈、堆、方法区
虚拟机栈和本地方法栈:每个线程创建时会创建一个虚拟机栈,如果调用的是本地方法则创建本地方法栈,当线程调用方法时会创建一个栈帧用于存放局部变量表、操作数栈、动态连接、方法出口等信息,方法的调用直至结束都对应着一个栈帧从出栈到入栈的过程,当线程申请的栈深度超过虚拟机允许的最大深度时会发生栈泄漏异常(StackOverflowError),如果栈深度是可扩展的,当线程无法为栈帧申请到内存时会发生内存溢出(OutOfMemoryError)异常。多层次的递归可能导致虚拟机栈内存泄漏的原因之一。
堆:堆中存放的是Java对象实例,可能的内存溢出的场景是java堆中已存在大量对象,这些对象保持着到GC Roots之间可访问的链路而避免被回收,从而导致虚拟机无法为新的对象实例分配内存。
方法区:方法区存放的是类型信息和常量,当大量动态生成类型或者生成大量运行时常量时可能导致方法区内存溢出。
另外还有一个内存区域是程序计数器,该区域用于存放线程执行的字节码的行号,Java虚拟机规范没有规定该区域存在内存溢出异常的情况。