概述
ClassLoader的主要职责就是负责加载各种class文件到JVM中,ClassLoader是一个抽象类。给定一个clss的二进制名,ClassLoader会尝试加载并且在JVM中生成构成这个类的的各个数据结构,然后使其分布在JVM对应的内存区域。
类的加载过程简介
类的加载过程一般分为三个比较大的阶段,分别是加载阶段,连接阶段和初始化阶段。
加载阶段:主要负责查找并且加载类的二进制数据文件,其实就是class文件
连接阶段:连接阶段还可以分为三个阶段
- 验证:主要是确保类文件的正确性,比如class的版本,class文件的魔术因子是否正确
- 准备: 为类的静态变量分配内存,并且为其初始化默认值
- 解析:把类中的符号引用转换为直接引用
初始化阶段:为类的静态变量赋予正确的初始值(代码编写阶段给定的值)
当一个JVM被我们启动时,包含的类非常多,但并不是每一个类都会被初始化。JVM对类的初始化是延迟机制,当一个类在首次主动使用时才会被初始化,在同一个运行时包下,一个Class只会被初始化一次
类的主动使用和被动使用
JVM虚拟机规范规定了,每个类或接口被Java程序首次主动使用时才会对其进行初始化。JVM同时规定了以下6种主动使用类的场景:
- 通过new的关键字导致类的初始化
- 访问类的静态变量导致类的初始化
- 调用静态方法导致类的初始化
- 对某个类进行反射操作导致类的初始化
- 初始化子类会导致父类的初始化
- 启动类:也就是执行main函数所在的类都会导致该类的初始化
除了上述6种情况,其余都称为被动使用,不会导致类的加载和初始化。下面几个栗子并不是主动使用,但是易混淆:
- 构造某个类的数组并不会导致该类的初始化
Simeple[] simples = new Simeple[10];
不要被new所混淆,实际上只是开辟了10*Simple大小的内存 - 引用类的常不会导致类的初始化,常量放在常量池中。
public class GlobleTest {
static {
System.out.println("类已被构造");
}
//在其他类中使用常量MAX不会导致GloableTest类被初始化,静态代码块不会执行
public final static int MAX=100;
//虽然random是静态常量,但是由于计算复杂,只有初始化之后才能得到结果
//因此在其他类中调用random,会导致GlobleTest类被初始化
public final static int random = new Random().nextInt();
}