JVM工作过程

将JVM工作过程粗略分为5个阶段,包括加载阶段、链接阶段、初始化阶段、执行阶段、回收阶段

其中,

(1)加载阶段、链接阶段的解析部分 主要由类加载器 完成

(2)初始化阶段是由JVM的类加载机制 在类加载过程的最后阶段自动触发 的。

(3)执行阶段主要由执行引擎 负责

(4)回收阶段主要是垃圾收集器(Garbage Collector)负责。

所以,在Java虚拟机(JVM)中,读取字节码文件、解析字节码文件为类信息,并将这些信息存储在运行时数据区的工作主要是由类加载器(ClassLoader)完成的。之后,执行引擎(Execution Engine)会利用这些存储在运行时数据区中的类信息(包括字节码),将其转换为特定平台上的机器码并执行。

1.加载阶段

类加载器负责将类的二进制数据(通常是.class文件)从文件系统、网络或其他来源读取 到JVM的内存中。这个过程包括:

(1)通过类的全限定名(Fully Qualified Name)来定位并读取 对应的字节码文件。

(2)将这个字节码文件所代表的静态存储结构转化 为方法区(在Java 8及以后版本中称为元空间Metaspace)的运行时数据结构。

(3)在内存中生成一个代表这个类的java.lang.Class对象,作为访问方法区中这些类数据的入口。

2.链接阶段

链接阶段包括验证、准备和解析三个子阶段,类加载器主要参与其中的验证和解析阶段(准备阶段不由类加载器直接负责,但也是类加载过程的一部分,在此提及以保持完整性)

(1)验证:验证是类加载过程的一部分,且是在加载阶段之后进行的。验证的目的是确保加载的字节码文件符合Java虚拟机规范 ,不会危害系统安全。验证包括文件格式验证、元数据验证、字节码验证和符号引用验证等。

(2)准备:为类的静态变量分配内存 ,并将其初始化为默认值 (如int类型的变量初始化为0,对象引用类型的变量初始化为null)。

(3)解析:将类中的符号引用(如字段名、方法名等)替换为直接引用(即内存地址或偏移量)。这一步是类加载器在链接阶段的主要工作之一,它使得JVM在运行时能够直接通过内存地址或偏移量来访问类的字段和方法。

3.初始化阶段

初始化阶段主要是执行类的初始化代码,包括静态代码块和静态变量的赋值操作。

(1)当类被加载、链接(包括验证、准备和解析)之后,如果该类被主动使用 (如创建类的实例、访问类的静态变量或静态方法等),JVM就会执行该类的初始化代码,即执行类构造器<clinit>()方法中的代码。

(2)类构造器<clinit>()方法是由编译器自动收集类中所有类变量的赋值动作和静态代码块(static block)中的语句合并产生的。它只会被执行一次,用于初始化类的静态变量和静态代码块。

(3)需要注意的是,初始化阶段与类的实例化(即创建类的对象)是不同的。类的实例化是通过调用类的构造器(<init>()方法,注意与类构造器<clinit>()区分)来完成的,而类的初始化则是在类被主动使用之前,由JVM自动触发的。

java 复制代码
public class InitializationExample {  
      
    // 静态变量初始化  
    static int staticVar = 42;  
      
    // 静态代码块  
    static {  
        System.out.println("Static block executed. staticVar = " + staticVar);  
        // 可以对静态变量进行修改  
        staticVar = staticVar * 2;  
    }  
  
    public static void main(String[] args) {  
        // 访问类的静态变量,触发类的初始化  
        System.out.println("Accessing static variable: staticVar = " + staticVar);  
  
        // 创建类的实例不会再次触发类的初始化,因为<clinit>()方法只执行一次  
        InitializationExample instance1 = new InitializationExample();  
        InitializationExample instance2 = new InitializationExample();  
  
        // 演示实例变量和构造器  
        instance1.instanceMethod();  
    }  
  
    // 实例变量  
    int instanceVar = 10;  
  
    // 构造器  
    public InitializationExample() {  
        System.out.println("Constructor called. instanceVar = " + instanceVar);  
    }  
  
    // 实例方法  
    public void instanceMethod() {  
        System.out.println("Instance method called. instanceVar = " + instanceVar);  
    }  
}

4.执行阶段

执行引擎可以采用多种技术来实现字节码的执行,主要包括解释执行即时编译 (JIT,Just-In-Time Compilation)两种方式。
解释执行 :解释器逐条读取字节码指令,并将其翻译成对应平台的机器码执行。这种方式实现简单,但执行效率相对较低,因为每次执行都需要进行翻译。
即时编译:JIT编译器将频繁执行的热点代码编译成目标平台的本地机器码,以提高执行效率。JVM通常会在程序运行时收集执行信息,识别出热点代码,并触发JIT编译。这种方式可以显著提高执行效率,但会增加编译的开销。

5.回收阶段

回收阶段主要是垃圾收集器 (Garbage Collector)负责的工作,它回收JVM中不再使用的对象和类。

参考:https://blog.csdn.net/mandy_shandong/article/details/139355123

相关推荐
何中应6 分钟前
Spring Boot中选择性加载Bean的几种方式
java·spring boot·后端
苏苏大大8 分钟前
zookeeper
java·分布式·zookeeper·云原生
wclass-zhengge35 分钟前
03垃圾回收篇(D3_垃圾收集器的选择及相关参数)
java·jvm
涛ing37 分钟前
23. C语言 文件操作详解
java·linux·c语言·开发语言·c++·vscode·vim
5xidixi1 小时前
Java TCP协议(2)
java·tcp/ip
2013crazy1 小时前
Java 基于 SpringBoot+Vue 的校园兼职平台(附源码、部署、文档)
java·vue.js·spring boot·兼职平台·校园兼职·兼职发布平台
小高不明1 小时前
仿 RabbitMQ 的消息队列3(实战项目)
java·开发语言·spring·rabbitmq·mybatis
兩尛1 小时前
订单状态定时处理、来单提醒和客户催单(day10)
java·前端·数据库
web2u1 小时前
MySQL 中如何进行 SQL 调优?
java·数据库·后端·sql·mysql·缓存
Yeats_Liao2 小时前
Spring 框架:配置缓存管理器、注解参数与过期时间
java·spring·缓存