Class装载系统
Class类型通常以文件的形式存在(当然任何二进制流都可以是Class类型),只有Java虚拟机装载的class文件才能在程序中使用,系统装载Class可以分为加载、连接、初始化。其中连接又可以分为验证、准备、解析。
class文件的装载流程如下:
类装载条件
Class只有在必须要使用的时候才会被装载,Java虚拟机不会无条件的装载Class类型。Java虚拟机规定,一个类或接口在初次使用时,必须要进行初始化。这里的使用是指“主动使用”,主动使用分为下面几种情况:
- 当创建一个类的实例时,比如使用new关键字,或者通过反射、克隆、反序列化
- 当调用类的静态方法时,即使用字节码invokestatic 指令
- 当使用类或接口的静态字段时(final常量除外),比如,使用getstatic 或者 putstatic指令。
- 当使用java.lang.reflect 包中的方法反射类的方法时。
- 当初始化子类时,要求先初始化父类
- 作为启动虚拟机,含有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虚拟机会执行以下操作
- 通过类的全名,获取类的二进制数据流
- 解析类的二进制数据流为方法区中的数据结构
- 创建java.lang.Class类的实例,表示该类型
验证类阶段
- 格式检查:魔数检查、版本检查、长度检查…
- 语义检查:是否有父类、是否继承final…
- 字节码验证:操作数类型是否合理、跳转指令是否指向正确位置…
- 符号引用验证:符合引用的直接引用是否存在…
准备阶段
当一个类通过验证时,虚拟机就会进入准备阶段,此时,会为这个类分配相应的内存空间,并设置初始值
类型 | 默认初始值 |
---|---|
int | 0 |
long | 0L |
short | (short)0 |
char | \u0000 |
boolean | false |
reference | null |
float | 0.0f |
double | 0.0d |
注意: Java并不支持Boolean类型,对于Boolean类型,内部实现是int,由于int默认值是0,故对于的Boolean默认值就是false
如果存在常量字段,那么常量字段也会在准备阶段被附上正确的值,这个复制是Java虚拟机的行为,属于变量的初始化,事实上,在准备阶段,不会有任何Java代码被执行。
解析阶段
将类、接口、字段和方法的符号引用转为直接引用
初始化阶段
这是类装载的最后一个阶段。如果前面步骤没有问题,那么类就可以正确装载到系统中。此时,类才会开始执行Java字节码