Java 虚拟机 (JVM) 是 Java 运行环境的核心部分,它负责执行 Java 字节码。JVM 由多个不同的组件组成,每个组件都负责不同的任务。下面我将详细介绍 JVM 的主要组成部分及其运行流程。
JVM 的组成部分
-
类加载器 (Class Loader):
- 类加载器负责读取文件系统中的
.class
文件,并将其转换成 Java 虚拟机可以理解的形式。 - 类加载器分为三个主要类型:启动类加载器(Bootstrap ClassLoader)、扩展类加载器(Extension ClassLoader)和应用程序类加载器(Application ClassLoader)。
- 类加载器负责读取文件系统中的
-
运行时常量池 (Runtime Constant Pool):
- 运行时常量池是方法区的一部分,它存放着类或接口编译期生成的各种字面量和符号引用。
-
方法区 (Method Area):
- 方法区是各个线程共享的内存区域,它用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码缓存等数据。
- Java 8 以后,方法区被称为"元空间"(Metaspace),它不再位于堆中,而是直接使用本地内存。
-
堆 (Heap):
- 堆是 Java 虚拟机所管理的内存中最大的一块,几乎所有的对象实例都在这里分配内存。
- 堆又被划分为新生代(Young Generation)和老年代(Old Generation),其中新生代又细分为 Eden 区、From Survivor 区和 To Survivor 区。
-
程序计数器 (Program Counter Register):
- 程序计数器是一块较小的内存空间,它可以看作是当前线程所执行的字节码的行号指示器。
-
虚拟机栈 (Virtual Machine Stack):
- 每个线程拥有一个独立的虚拟机栈,栈中存储的是栈帧。
- 栈帧是方法执行的上下文信息,包括局部变量表、操作数栈、动态链接、方法出口等信息。
-
本地方法栈 (Native Method Stack):
- 与虚拟机栈相似,但是本地方法栈为虚拟机使用到的 Native 方法服务。
-
垃圾回收器 (Garbage Collector):
- 垃圾回收器负责回收不再使用的对象所占用的内存空间,以减少内存泄漏的可能性。
-
执行引擎 (Execution Engine):
- 执行引擎负责解释执行字节码或执行由 JIT 编译器编译的本地代码。
-
编译器 (Compiler):
- 包括即时编译器(JIT Compiler),它将频繁执行的热点代码编译成本地机器码,以提高性能。
JVM 的运行流程
-
加载类:
- 当 Java 程序启动时,类加载器会加载主类及其依赖的类。
- 类加载过程包括加载、验证、准备、解析和初始化五个阶段。
-
创建线程:
- 创建主线程(main thread),执行
main
方法。
- 创建主线程(main thread),执行
-
执行:
- 主线程开始执行
main
方法,调用其他方法或创建新的线程。 - 每个线程都有自己的虚拟机栈和程序计数器。
- 主线程开始执行
-
分配内存:
- 对象实例在堆中分配内存。
- 局部变量和方法参数在虚拟机栈中分配内存。
-
执行字节码:
- 执行引擎负责解释执行字节码。
- 热点代码会被 JIT 编译器编译成本地代码以提高性能。
-
垃圾回收:
- 当内存不足时,垃圾回收器会被触发,回收不再使用的对象所占用的空间。
-
结束:
- 当所有线程执行完毕或者程序正常退出时,JVM 将终止。
总结
JVM 是一个复杂的软件系统,它不仅负责执行 Java 字节码,还管理内存、处理线程调度、执行垃圾回收等。