JVM加载机制

1、JVM加载机制

Class类型通常以文件的形式存在(当然,任何二进制流都可以是Class类型的),只有被Java虚拟机装载的Class类型才能在程序中使用。

类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个java.lang.Class对象,用来封装类在方法区内的数据结构。类的加载的最终产品是位于堆区中的Class对象,Class对象封装了类在方法区内的数据结构,并且向Java程序员提供了访问方法区内的数据结构的接口。

2、加载步骤

加载(Loading)

这是类加载过程的第一个阶段。在这一阶段,JVM通过类的全名(包括包名)找到并加载.class文件。这些.class文件可以来自文件系统、网络或其他来源。

加载完成后,JVM会在内存中为这个类创建一个对应的java.lang.Class对象,这个对象作为该类所有信息的入口点。

链接(Linking)

链接阶段又可以分为验证、准备和解析三个子阶段。

  • 验证(Verification):验证阶段确保被加载的类的正确性和安全性。它会检查类的字节码是否符合JVM规范,以及是否有潜在的安全风险。
  • 准备(Preparation):准备阶段为类的静态变量分配内存,并设置初始值。注意,这里的初始值是Java虚拟机规范所规定的默认值,而不是在代码中显示的初始值。
  • 解析(Resolution):解析阶段是将常量池内的符号引用转换为直接引用的过程。这涉及到对类、接口、字段和方法的符号引用到实际内存地址的映射。

初始化(Initialization)

初始化阶段是类加载过程的最后一步。在这一阶段,JVM执行类的构造器方法()。这个方法是由编译器自动收集类中的所有类变量的赋值动作和静态代码块(static{}块)中的语句合并产生的。
如果一个类有父类,那么在初始化这个类之前,会先初始化其父类。这是为了确保父类中的静态变量和静态代码块在子类使用之前已经被正确初始化。

以下是一个简单的Java程序,演示了类加载的过程:

java 复制代码
public class MyClass {
    static int staticVariable = 10;
    static final int FINAL_STATIC_VAR = 20;
 
    static {
        System.out.println("MyClass static block executed");
    }
}
 
public class Loader {
    public static void main(String[] args) throws ClassNotFoundException {
        // 通过这种方式,我们可以触发类的装载和链接
        Class<?> clazz = Class.forName("MyClass");
 
        // 打印装载和链接过程中的一些信息
        System.out.println("Class " + clazz.getName() + " has been loaded");
 
        // 访问类的静态字段,将触发初始化
        System.out.println("Static variable value: " + MyClass.staticVariable);
        System.out.println("Final static variable value: " + MyClass.FINAL_STATIC_VAR);
    }
}

当运行Loader类的main方法时,JVM将会加载MyClass类,这将触发链接和初始化过程。Class.forName("MyClass")这行代码会装载并链接MyClass类,然后初始化它。在初始化过程中,会执行静态初始化块,并为静态变量赋值。

输出将会是:

java 复制代码
MyClass static block executed
Class MyClass has been loaded
Static variable value: 10
Final static variable value: 20

此外,JVM的类加载机制还遵循双亲委派模型。当一个类加载器需要加载一个类时,它首先会把这个请求委派给父类加载器去完成,每一个层次的类加载器都是如此,因此所有的加载请求最终都应该传送到顶层的启动类加载器中。只有当父类加载器无法完成这个加载请求(它的搜索范围中没有找到所需的类)时,子加载器才会尝试自己去加载。这种机制有助于确保Java应用程序的稳定性和安全性。

需要注意的是,类加载的触发时机并不是随意的,而是在读取或设置一个类的静态变量、对类进行反射调用、虚拟机启动时定义了main()方法的那个类先初始化等特定情况下才会发生。

总的来说,JVM的类加载机制是一个复杂而精巧的系统,它确保了Java程序的正确性和安全性,并提供了"一次编写,到处运行"的能力。

对于java 无参、有参构造方法;静态、非静态代码块执行顺序

可参考另一篇文章
https://blog.csdn.net/qiyeliuli/article/details/54867055

相关推荐
Doris_20232 分钟前
Python 模式匹配match case
前端·后端·python
每天回答3个问题14 分钟前
UE5C++编译遇到MSB3073
开发语言·c++·ue5
伍哥的传说27 分钟前
Vite Plugin PWA – 零配置构建现代渐进式Web应用
开发语言·前端·javascript·web app·pwa·service worker·workbox
这里有鱼汤28 分钟前
Python量化实盘踩坑指南:分钟K线没处理好,小心直接亏钱!
后端·python·程序员
小莞尔1 小时前
【51单片机】【protues仿真】 基于51单片机八路抢答器系统
c语言·开发语言·单片机·嵌入式硬件·51单片机
大模型真好玩1 小时前
深入浅出LangGraph AI Agent智能体开发教程(五)—LangGraph 数据分析助手智能体项目实战
人工智能·python·mcp
测试老哥1 小时前
Selenium 使用指南
自动化测试·软件测试·python·selenium·测试工具·职场和发展·测试用例
我是菜鸟0713号1 小时前
Qt 中 OPC UA 通讯实战
开发语言·qt
JCBP_1 小时前
QT(4)
开发语言·汇编·c++·qt·算法
码熔burning1 小时前
JVM 垃圾收集算法详解!
jvm·算法