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

相关推荐
Cpdr2 分钟前
pytorch自适应的调整特征图大小
pytorch·python·深度学习
写代码的中青年5 分钟前
Semantic Kernel:微软大模型开发框架——LangChain 替代
人工智能·python·microsoft·langchain·大模型·llm
ZJ_.9 分钟前
Node.js 使用 gRPC:从定义到实现
java·开发语言·javascript·分布式·rpc·架构·node.js
ljh_a122 分钟前
Django 和 Django REST framework 创建对外 API
python·http·django·flask·tornado
syluxhch23 分钟前
Pycharm的终端(Terminal)中切换到当前项目所在的虚拟环境
ide·python·pycharm
yjjpp230124 分钟前
Django REST Framework(四)DRF APIVIEW
后端·python·django
concisedistinct28 分钟前
大数据开发语言 Scala(四):面向对象编程
大数据·开发语言·后端·scala·编程语言·面向对象
铁匠匠匠28 分钟前
django学习入门系列之第三点《BootSrap初了解》
前端·经验分享·笔记·python·学习·django·前端框架
_Rookie._35 分钟前
java 单例模式
java·开发语言·单例模式