Java 类加载过程

待我称王封你为后i 2023-01-23 10:54 60阅读 0赞

一、加载

加载一个Class需要完成以下3件事:

  • 通过Class的全限定名获取Class的二进制字节流
  • 将Class的二进制内容加载到虚拟机的方法区
  • 在内存中生成一个java.lang.Class对象表示这个Class

获取Class的二进制字节流这个步骤有多种方式:

  • 从zip中读取,如:从jar、war、ear等格式的文件中读取Class文件内容
  • 从网络中获取,如:Applet
  • 动态生成,如:动态代理、ASM框架等都是基于此方式
    由其他文件生成,典型的是从jsp文件生成相应的Class

二、校验

验证一个Class的二进制内容是否合法,主要包括4个阶段:

  • 文件格式验证,确保文件格式符合Class文件格式的规范。如:验证魔数、版本号等。
  • 元数据验证,确保Class的语义描述符合Java的Class规范。如:该Class是否有父类、是否错误继承了final类、是否一个合法的抽象类等。
  • 字节码验证,通过分析数据流和控制流,确保程序语义符合逻辑。如:验证类型转换是合法的。
  • 符号引用验证,发生于符号引用转换为直接引用的时候(转换发生在解析阶段)。如:验证引用的类、成员变量、方法的是否可以被访问(IllegalAccessError),当前类是否存在相应的方法、成员等(NoSuchMethodError、NoSuchFieldError)。

三、准备

在准备阶段,虚拟机会在方法区中为Class分配内存,并设置static成员变量的初始值为默认值。注意这里仅仅会为static变量分配内存(static变量在方法区中),并且初始化static变量的值为其所属类型的默认值。如:int类型初始化为0,引用类型初始化为null。即使声明了这样一个static变量:

public static int a = 123;

在准备阶段后,a在内存中的值仍然是0, 赋值123这个操作会在中初始化阶段执行,因此在初始化阶段产生了对应的Class对象之后a的值才是123 。

四、解析

解析阶段,虚拟机会将常量池中的符号引用替换为直接引用,解析主要针对的是类、接口、方法、成员变量等符号引用。在转换成直接引用后,会触发校验阶段的符号引用验证,验证转换之后的直接引用是否能找到对应的类、方法、成员变量等。这里也可见类加载的各个阶段在实际过程中,可能是交错执行

五、初始化

初始化阶段即开始在内存中构造一个Class对象来表示该类,即执行类构造器<clinit>()的过程。需要注意下,<clinit>()不等同于创建类实例的构造方法<init>()

<clinit>()方法中执行的是对static变量进行赋值的操作,以及static语句块中的操作。
虚拟机会确保先执行父类的<clinit>()方法。
如果一个类中没有static的语句块,也没有对static变量的赋值操作,那么虚拟机不会为这个类生成<clinit>()方法。
虚拟机会保证<clinit>()方法的执行过程是线程安全的。
因此,存在如下一种最简单的单例模式的实现:

  1. public class Singleton {
  2. public static final INSTANCE = new Singleton();
  3. private Singleton() {
  4. }
  5. }

参考资料:《深入理解Java虚拟机》

发表评论

表情:
评论列表 (有 0 条评论,60人围观)

还没有评论,来说两句吧...

相关阅读

    相关 Java过程

    目录 概述 类加载时机 类加载过程 加载 验证 文件格式验证 元数据验证 字节码校验 符号引用校验 准备 解析 初始化 类加载器 双亲委派模型

    相关 Java过程

    一、前言 我们都知道计算机的底层逻辑都是0和1的编码,当然除了现在所研究的量子计算除外。那么我们在计算机所做的一切操作,底层原理是不是都可以翻译到0和1呢。如果刨根问底的话,

    相关 Java 过程

    在Java中,类装载器把一个类装入Java虚拟机中,要经过三个步骤来完成:装载、链接和初始化,其中链接又可以分成校验、准备、解析 装载:查找和导入类或接口的二进制数据;

    相关 Java过程

    类的加载简介 1.类加载一般分为三大阶段:加载、连接和初始化 ![类加载三个大阶段(图来自zejian\_)][zejian] 加载阶段:主要查找class文件(二

    相关 Java过程

    Java类加载过程 首先是加载过程(Loading),它是 Java 将字节码数据从不同的数据源读取到 JVM 中,并映射为 JVM 认可的数据结构(Class 对象),...