程序计数器
线程私有,可以看作当前线程所执行的字节码的行号指示器。字节码解释器工作时通过 改变这个计数器的值来选取吓一跳需要执行的字节码指令。因为每次线程切换后需要保存当前执行到的位置,所以每个线程都需要有一个程序计数器。若当前线程执行的是一个Java方法,则记录正在执行的虚拟机字节码指向的地址;若执行的是Native方法,则为空。此区域是唯一没有OOM情况的内存区域。
Java虚拟机栈
线程私有,生命周期与线程相同。其中保存的是栈帧。每个Java方法在执行时会创建一个栈帧,用于保存局部变量表、操作数栈、动态链接、方法出口等信息。每个方法从调用到执行完成的过程对应着一个栈帧在虚拟机栈中的入栈到出栈的过程。局部变量表:存放编译器可知的各种基本数据类型(8种)、对象引用类型(仅仅是引用而不是对象的实例)和returnAddress类型(指向了一条字节码执行的位置)。若线程请求的栈深度大于虚拟机所允许的深度将抛出OutOfMemoryError,虚拟机栈可以动态扩展时若无法申请到足够的内存则抛出OutOfMemoryError。
本地方法栈
线程私有,与虚拟机栈非常相似。虚拟机栈为Java方法服务,而本地方法栈为Native方法服务。Native方法用非Java语言实现。
Java堆
占据内存中最大的一块,线程共享。用于存放对象实例,几乎所有的对象实例都在这里分配内存。也称为GC堆,即为垃圾收集器管理的主要区域。可细分为新生代和老年代,新生代可分为Eden区、FromSurvivor区和ToSurvivor区,用于存放不同年龄的对象实例,便于垃圾回收。若堆中没有内存完成实例分配也无法扩展时,将会抛出OutOfMemoryError。
方法区
线程共享,用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。此区域的内存回收目标主要是针对常量池的回收和对类型的卸载,一般效率不高。无法满足内存分配需求也无法扩展时抛出OutOfMemoryError。