罗罗解析Class文件的装载流程

Class装载系统

Class类型通常以文件的形式存在(当然任何二进制流都可以是Class类型),只有Java虚拟机装载的class文件才能在程序中使用,系统装载Class可以分为加载、连接、初始化。其中连接又可以分为验证、准备、解析。

class文件的装载流程如下:
在这里插入图片描述
类装载条件

Class只有在必须要使用的时候才会被装载,Java虚拟机不会无条件的装载Class类型。Java虚拟机规定,一个类或接口在初次使用时,必须要进行初始化。这里的使用是指“主动使用”,主动使用分为下面几种情况:

  1. 当创建一个类的实例时,比如使用new关键字,或者通过反射、克隆、反序列化
  2. 当调用类的静态方法时,即使用字节码invokestatic 指令
  3. 当使用类或接口的静态字段时(final常量除外),比如,使用getstatic 或者 putstatic指令。
  4. 当使用java.lang.reflect 包中的方法反射类的方法时。
  5. 当初始化子类时,要求先初始化父类
  6. 作为启动虚拟机,含有main方法的那个类

除了以上情况属于主动使用,其他情况均属于被动使用。被动使用不会引起类的初始化。

示例主动引用

public class Parent {
    static {
        System.out.println("parent  init...");
    }
}
public class Child extends Parent {
    static {
        System.out.println("child init...");
    }

}
public class test02 {
    public static void main(String[] args) {
        Child child = new Child();
    }
}

输出

"C:\Program Files\Java\jdk1.8.0_144\bin\java.exe" "-javaagent:F:\MyDir\IDEA\IDEA2018\IntelliJ IDEA 2018.2.4\lib\idea_rt.jar=50620:F:\MyDir\IDEA\IDEA2018\IntelliJ IDEA Files\Java\jdk1.8.0_144\jre\lib\rt.jar;E:\idea_workplace\javaweb\target\classes;C:\Users\V5459\.m2\repository\mysql\mysql-connector-java\5.1.6\mysql-connector-java-5.1.6.jar;C:\Users\V5459\.m2\repository\junit\junit\4.11\junit-4.11.jar;C:\Users\V5459\.m2\repository\org\hamcrest\hamcrest-core\1.3\hamcrest-core-1.3.jar" 
parent  init...
child init...

Process finished with exit code 0

由此可知,先装载的是Parent类,然后才是Child类。符合两个主动装载中的两个条件,即使用了new关键字,在初始化子类时,必须先初始化父类。

下面看看被动引用的例子

public class Parent {
    static {
        System.out.println("parent  init...");
    }
    public static int num = 200;
}
public class Child extends Parent {
    static {
        System.out.println("child init...");
    }

}
public class test02 {
    public static void main(String[] args) {
        System.out.println(Child.num);
    }
}
"C:\Program Files\Java\jdk1.8.0_144\bin\java.exe" "-javaagent:F:\MyDir\IDEA\IDEA2018\IntelliJ IDEA 2018.2.4\lib\idea_rt.jar=50771:F:\MyDir\IDEA\IDEA2018\IntelliJ IDEA Files\Java\jdk1.8.0_144\jre\lib\rt.jar;E:\idea_workplace\javaweb\target\classes;C:\Users\V5459\.m2\repository\mysql\mysql-connector-java\5.1.6\mysql-connector-java-5.1.6.jar;C:\Users\V5459\.m2\repository\junit\junit\4.11\junit-4.11.jar;C:\Users\V5459\.m2\repository\org\hamcrest\hamcrest-core\1.3\hamcrest-core-1.3.jar"
parent  init...
200

Process finished with exit code 0

可见,Parent被初始化,而Child未被初始化,虽然Child类未被初始化,但却是被加载进系统了的。
在引用一个字段时,只有直接定义该字段的类,才会被初始化。

使用 -XX:+TraceClassLoading 参数运行,就会得到下面日志(只列出了部分输出)

...
[Loaded java.lang.Void from C:\Program Files\Java\jdk1.8.0_144\jre\lib\rt.jar]
[Loaded sun.nio.cs.US_ASCII$Decoder from C:\Program Files\Java\jdk1.8.0_144\jre\lib\rt.jar]
[Loaded jdbctest.Parent from file:/E:/idea_workplace/javaweb/target/classes/]
[Loaded jdbctest.Child from file:/E:/idea_workplace/javaweb/target/classes/]
parent  init...
200
[Loaded java.lang.Shutdown from C:\Program Files\Java\jdk1.8.0_144\jre\lib\rt.jar]
[Loaded java.lang.Shutdown$Lock from C:\Program Files\Java\jdk1.8.0_144\jre\lib\rt.jar]

Process finished with exit code 0

从日志中可以看到,Child类的确是被加载进系统,但是初始化却未进行。

还有一种很特殊,我上面说过,当使用类或接口的静态字段时(final常量除外)。也就是说引用final常量,并不会引起类的初始化。下面来看看这个特殊的final

public class Parent {
    static {
        System.out.println("parent  init...");
    }
    public static final int num = 200;
}

public class test02 {
    public static void main(String[] args) {
        System.out.println(Parent.num);
    }
}
"C:\Program Files\Java\jdk1.8.0_144\bin\java.exe" "-javaagent:F:\MyDir\IDEA\IDEA2018\IntelliJ IDEA 2018.2.4\lib\idea_rt.jar=50620:F:\MyDir\IDEA\IDEA2018\IntelliJ IDEA Files\Java\jdk1.8.0_144\jre\lib\rt.jar;E:\idea_workplace\javaweb\target\classes;C:\Users\V5459\.m2\repository\mysql\mysql-connector-java\5.1.6\mysql-connector-java-5.1.6.jar;C:\Users\V5459\.m2\repository\junit\junit\4.11\junit-4.11.jar;C:\Users\V5459\.m2\repository\org\hamcrest\hamcrest-core\1.3\hamcrest-core-1.3.jar" 
200

Process finished with exit code 0

结果表示,Parent类并没有被加载进系统

看下日志

[Loaded java.lang.Class$MethodArray from C:\Program Files\Java\jdk1.8.0_144\jre\lib\rt.jar]
[Loaded java.lang.Void from C:\Program Files\Java\jdk1.8.0_144\jre\lib\rt.jar]
200
[Loaded java.lang.Shutdown from C:\Program Files\Java\jdk1.8.0_144\jre\lib\rt.jar]
[Loaded java.lang.Shutdown$Lock from C:\Program Files\Java\jdk1.8.0_144\jre\lib\rt.jar]

Process finished with exit code 0

这是由于Class文件在生成时,final常量由于其不变性,做了适当的优化。将final常量直接放入常量池中。

加载类阶段

加载类处于类加载的第一个阶段,加载类时,Java虚拟机会执行以下操作

  1. 通过类的全名,获取类的二进制数据流
  2. 解析类的二进制数据流为方法区中的数据结构
  3. 创建java.lang.Class类的实例,表示该类型
验证类阶段
  1. 格式检查:魔数检查、版本检查、长度检查…
  2. 语义检查:是否有父类、是否继承final…
  3. 字节码验证:操作数类型是否合理、跳转指令是否指向正确位置…
  4. 符号引用验证:符合引用的直接引用是否存在…
准备阶段

当一个类通过验证时,虚拟机就会进入准备阶段,此时,会为这个类分配相应的内存空间,并设置初始值

类型默认初始值
int0
long0L
short(short)0
char\u0000
booleanfalse
referencenull
float0.0f
double0.0d

注意: Java并不支持Boolean类型,对于Boolean类型,内部实现是int,由于int默认值是0,故对于的Boolean默认值就是false

如果存在常量字段,那么常量字段也会在准备阶段被附上正确的值,这个复制是Java虚拟机的行为,属于变量的初始化,事实上,在准备阶段,不会有任何Java代码被执行。

解析阶段

将类、接口、字段和方法的符号引用转为直接引用

初始化阶段

这是类装载的最后一个阶段。如果前面步骤没有问题,那么类就可以正确装载到系统中。此时,类才会开始执行Java字节码

  • 3
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

罗罗的1024

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值