Java教程

Java控制语句

面向对象编程

Java内置类

Java文件处理

Java错误与异常

Java多线程

Java同步

Java网络编程

Java集合

Java接口

Java数据结构

Java集合算法

高级Java

Java杂项

Java APIs与框架

Java类参考

Java实用资源

JVM(Java虚拟机)架构



什么是JVM(Java虚拟机)?

JVM(Java虚拟机)是一个虚拟机,一个抽象的计算机,拥有自己的指令集架构(ISA)、内存、堆栈、堆等。它运行在主机操作系统上,并向其提出资源需求。

JVM(Java虚拟机)是一个规范,只要符合规范,就可以有不同的实现。规范可以在以下链接中找到:https://docs.oracle.com

Oracle有自己的JVM实现(称为HotSpot JVM),IBM也有自己的(例如J9 JVM)。

规范中定义的操作如下所示(来源 - Oracle JVM规范)

  • “.class”文件格式
  • 数据类型
  • 基本类型和值
  • 引用类型和值
  • 运行时数据区域
  • 栈帧
  • 对象的表示
  • 浮点运算
  • 特殊方法
  • 异常
  • 指令集摘要
  • 类库
  • 公共设计,私有实现

JVM(Java虚拟机)架构

HotSpot JVM 3 的架构如下所示:

Architecture

执行引擎包含垃圾收集器和JIT编译器。JVM有两种类型:客户端服务器端。两者共享相同的运行时代码,但使用的JIT不同。稍后我们将详细了解这一点。用户可以通过指定JVM标志 -client 或 -server 来控制使用哪种类型。服务器端JVM是为服务器上长期运行的Java应用程序设计的。

JVM有32位和64位版本。用户可以使用VM参数中的-d32或-d64来指定要使用的版本。32位版本最多只能寻址4G内存。对于在内存中维护大型数据集的关键应用程序,64位版本可以满足这一需求。

JVM(Java虚拟机)架构的组成部分

以下是JVM(Java虚拟机)架构的主要组成部分:

1. 类加载器

JVM以动态的方式管理加载、链接和初始化类和接口的过程。在加载过程中,JVM查找类的二进制表示并创建它。

在链接过程中,加载的类被组合到JVM的运行时状态中,以便它们可以在初始化阶段执行。JVM基本上使用存储在类的方法区域的运行时常量池中的符号表来进行链接过程。初始化包括实际执行链接的类

以下是类加载器的类型:

  • 引导类加载器:此类加载器位于类加载器层次结构的顶部。它加载JRE的lib目录中的标准JDK类。
  • 扩展类加载器:此类加载器位于类加载器层次结构的中间,是引导类加载器的直接子类,加载JRE的lib\ext目录中的类。
  • 应用程序类加载器:此类加载器位于类加载器层次结构的底部,是应用程序类加载器的直接子类。它加载由CLASSPATH ENV变量指定的jar和类。

2. 链接和初始化

链接过程包含以下三个步骤:

  • 验证 - 这由字节码验证器完成,以确保生成的.class文件(字节码)有效。如果不是,则抛出错误,链接过程停止。
  • 准备 - 为类的所有静态变量分配内存,并使用默认值初始化它们。
  • 解析 - 所有符号内存引用都替换为原始引用。为此,使用了类的方法区域中运行时常量内存中的符号表。

初始化是类加载过程的最后阶段。静态变量被赋予原始值,并执行静态块。

3. 运行时数据区域

JVM规范定义了程序执行期间需要的某些运行时数据区域。其中一些在JVM启动时创建。其他区域是特定于线程的,只有在创建线程时才创建(并在线程销毁时销毁)。这些列在下面:

PC(程序计数器)寄存器

它特定于每个线程,并包含线程当前正在执行的JVM指令的地址。

它特定于每个线程,在方法调用期间存储参数、局部变量和返回地址。如果线程请求的堆栈空间超过允许的范围,则可能会发生StackOverflowError。如果堆栈是动态可扩展的,它仍然可以抛出OutOfMemoryError。

它被所有线程共享,包含在运行时创建的对象、类的元数据、数组等。它在 JVM 启动时创建,在 JVM 关闭时销毁。您可以使用某些标志控制 JVM 向操作系统请求的堆大小(稍后详细介绍)。需要注意的是,不要请求过少或过多的内存,因为它会严重影响性能。此外,GC 管理此空间,并持续删除不再使用的对象以释放空间。

方法区

这个运行时区域对所有线程都是通用的,在 JVM 启动时创建。它存储每个类的结构,例如常量池(稍后详细介绍)、构造函数和方法的代码、方法数据等。《Java 语言规范》(JLS) 没有规定此区域是否需要进行垃圾回收,因此,JVM 的实现可以选择忽略 GC。此外,它可以根据应用程序的需求进行扩展或不扩展。JLS 对此没有任何强制规定。

运行时常量池

JVM 维护一个每个类/每个类型的、用作符号表(它的众多角色之一)的数据结构,用于链接已加载的类。

本地方法栈

当线程调用本地方法时,它会进入一个新的环境,在这个环境中,Java 虚拟机的结构和安全限制不再限制它的自由。本地方法可能会访问虚拟机的运行时数据区域(取决于本地方法接口),但也可能执行任何其他操作。

4. 执行引擎

执行引擎负责执行字节码,它包含三个不同的组件

垃圾回收

JVM 管理 Java 中对象的整个生命周期。一旦创建了对象,开发人员就不必再担心它。如果对象变成不再使用(即,不再有对它的引用),它将由 GC 使用多种算法(串行 GC、CMS、G1 等)从堆中移除。

另请阅读: Java 中的垃圾回收

在 GC 过程中,对象会在内存中移动。因此,在该过程进行期间,这些对象不可用。整个应用程序必须在该过程持续时间内停止。此类暂停称为“停止世界”暂停,并且开销巨大。GC 算法的主要目标是减少此时间。

解释器

解释器解释字节码。它解释代码的速度很快,但执行速度很慢。

JIT 编译器

JIT 代表 Just-In-Time(即时)。JIT 编译器是 Java 运行时环境的主要部分,它在运行时将字节码编译为机器码。

5. Java 本地接口 (JNI)

Java 本地接口 (JNI) 与对执行必不可少的本地方法库交互。

6. 本地方法库

本地方法库是 C 和 C++ 库(本地库)的集合,对执行必不可少。

广告