JVM(Java Virtual Machine,Java 虚拟机)的内存区域划分是理解 Java 程序运行机制和内存管理的关键内容,不同区域有着各自特定的功能和作用,以下按照《Java 虚拟机规范》详细介绍:
一、程序计数器(Program Counter Register)
- 作用:它是一块较小的内存空间,可看作是当前线程所执行的字节码的行号指示器 。字节码解释器工作时,通过改变这个计数器的值来选取下一条需要执行的字节码指令,比如分支、循环、跳转、异常处理、线程恢复等操作都依赖它来完成流程控制 。
- 线程私有性:为了线程切换后能恢复到正确的执行位置,每条线程都需要有独立的程序计数器,各线程之间计数器互不影响,独立存储,所以是线程私有的内存区域 。
- 特殊情况:如果线程正在执行的是 Java 方法,这个计数器记录的是正在执行的虚拟机字节码指令的地址;如果正在执行的是本地(Native)方法,计数器值则为空(Undefined) 。此区域是唯一一个在《Java 虚拟机规范》中没有规定 OutOfMemoryError 情况的区域 。
二、Java 虚拟机栈(Java Virtual Machine Stacks)
- 作用:描述的是 Java 方法执行的线程内存模型,每个方法被执行的时候,Java 虚拟机都会同步创建一个栈帧(Stack Frame)用于存储局部变量表、操作数栈、动态连接、方法出口等信息 。每一个方法从调用直至执行完成的过程,就对应着一个栈帧在虚拟机栈中入栈到出栈的过程 。
- 线程私有性:与程序计数器类似,Java 虚拟机栈也是线程私有的,它的生命周期与线程相同 。
- 局部变量表:用于存放编译期可知的各种基本数据类型(boolean、byte、char、short、int、float、long、double )、对象引用(reference 类型,它不等同于对象本身,可能是一个指向对象起始地址的引用指针,也可能是指向一个代表对象的句柄或者其他与对象相关的位置 )和 returnAddress 类型(指向了一条字节码指令的地址 )。局部变量表所需的内存空间在编译期间完成分配,当进入一个方法时,这个方法需要在栈帧中分配多大的局部变量空间是完全确定的,在方法运行期间不会改变局部变量表的大小 。
- 异常情况:如果线程请求的栈深度大于虚拟机所允许的深度,将抛出 StackOverflowError 异常;如果 Java 虚拟机栈容量可以动态扩展(HotSpot 虚拟机的 Java 虚拟机栈就可以动态扩展,不过也可以通过 -Xss 参数预先设置固定大小 ),当扩展时无法申请到足够的内存,就会抛出 OutOfMemoryError 异常 。
三、本地方法栈(Native Method Stacks)
- 作用:与 Java 虚拟机栈作用类似,区别是 Java 虚拟机栈为虚拟机执行 Java 方法(也就是字节码)服务,而本地方法栈则是为虚拟机使用到的本地(Native)方法服务 。
- 线程私有性:同样是线程私有的内存区域,生命周期与线程一致 。
- 实现与异常:本地方法栈的具体实现可以由虚拟机自行规定,在 HotSpot 虚拟机中,本地方法栈和 Java 虚拟机栈是合二为一的 。当线程请求的本地方法栈深度溢出或者栈扩展无法申请到足够内存时,会分别抛出 StackOverflowError 异常和 OutOfMemoryError 异常 ,与 Java 虚拟机栈类似 。
四、Java 堆(Java Heap)
- 作用:是 Java 虚拟机所管理的内存中最大的一块,被所有线程共享的内存区域,在虚拟机启动时创建 。此区域的唯一目的就是存放对象实例,几乎所有的对象实例(包括数组,不过也有特殊情况,比如逃逸分析优化后,对象可能会分配在栈上 )以及数组都在这里分配内存 ,是垃圾收集器管理的主要区域,因此也常被称做 “GC 堆”(Garbage Collected Heap ) 。
- 内存划分:现代垃圾收集器基本都采用分代收集算法,所以 Java 堆还可以细分为:新生代和老年代;新生代又可以分为 Eden 空间、From Survivor 空间、To Survivor 空间等 。这样划分是为了更好地配合垃圾收集器工作,不同分代有不同的垃圾回收策略和时机 。
- 内存扩展与异常:Java 堆可以处于物理上不连续的内存空间中,只要逻辑上是连续的即可,就像我们的磁盘空间一样 。它的大小可以通过 -Xmx(最大堆大小 )和 -Xms(初始堆大小 )等参数进行控制 。如果堆中没有内存完成实例分配,并且堆也无法再扩展时,将会抛出 OutOfMemoryError 异常 。
五、方法区(Method Area)
- 作用:与 Java 堆一样,是各个线程共享的内存区域,它用于存储已被虚拟机加载的类型信息(类的全限定名、父类全限定名、接口列表、字段描述、方法描述等 )、常量、静态变量、即时编译器编译后的代码缓存等数据 。在 JDK 8 之前,方法区的实现常被称为 “永久代”(Permanent Generation ),不过这只是 HotSpot 虚拟机的一个实现选择,并非《Java 虚拟机规范》的强制要求;在 JDK 8 及以后,HotSpot 虚拟机取消了永久代,改用元空间(MetaSpace )来实现方法区,元空间使用的是本地内存 。
- 运行时常量池:是方法区的一部分,Class 文件中除了有类的版本、字段、方法、接口等描述信息外,还有一项信息是常量池表(Constant Pool Table ),用于存放编译期生成的各种字面量和符号引用,这部分内容将在类加载后存放到方法区的运行时常量池中 。运行时常量池相对于 Class 文件常量池的另外一个重要特征是具备动态性,Java 语言并不要求常量一定只有编译期才能产生,也就是并非预置入 Class 文件中常量池的内容才能进入方法区运行时常量池,运行期间也可以将新的常量放入池中,比如 String 类的 intern() 方法就可以在运行时把字符串常量放入运行时常量池(JDK 1.7 及以后字符串常量池移到堆中,不过 intern 机制等相关逻辑仍与方法区运行时常量池关联 ) 。
- 内存回收与异常:方法区的内存回收目标主要是针对常量池的回收和对类型的卸载,不过一般来说回收效果比较难令人满意,尤其是类型的卸载,条件相当苛刻,但这部分区域的回收确实是必要的 。当方法区无法满足内存分配需求时,将抛出 OutOfMemoryError 异常 。
以上这些内存区域共同构成了 JVM 运行时的数据区域,不同区域在 Java 程序执行过程中承担着不同职责,协同保障程序的正常运行和内存的有效管理 。