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

相关推荐
没有梦想的咸鱼185-1037-166318 分钟前
基于R语言机器学习方法在生态经济学领域中的实践技术应用
开发语言·机器学习·数据分析·r语言
2401_8288906421 分钟前
使用 BERT 实现意图理解和实体识别
人工智能·python·自然语言处理·bert·transformer
向上的车轮43 分钟前
基于go语言的云原生TodoList Demo 项目,验证云原生核心特性
开发语言·云原生·golang
The Chosen One98544 分钟前
C++ : AVL树-详解
开发语言·c++
PH_modest1 小时前
【Qt跬步积累】—— 初识Qt
开发语言·qt
BillKu1 小时前
Java核心概念详解:JVM、JRE、JDK、Java SE、Java EE (Jakarta EE)
java·jvm·jdk·java ee·jre·java se·jakarta ee
怀旧,1 小时前
【C++】18. 红⿊树实现
开发语言·c++
多恩Stone2 小时前
【3DV 进阶-2】Hunyuan3D2.1 训练代码详细理解下-数据读取流程
人工智能·python·算法·3d·aigc
xiaopengbc2 小时前
在 Python 中实现观察者模式的具体步骤是什么?
开发语言·python·观察者模式
Python大数据分析@2 小时前
python用selenium怎么规避检测?
开发语言·python·selenium·网络爬虫