类加载
1 基本说明
反射机制是 Java 实现动态语言的关键,也就是通过反射实现类动态加载。
(1)静态加载:编译时加载相关的类,如果没有则报错,依赖性太强;
(2)动态加载:运行时加载需要的类,如果运行时不用该类,即使不存在该类,也不报错,降低了依赖性;
2 类加载时机
(1)当创建对象时(new)//静态加载
(2)当子类被加载时,父类也加载 //静态加载
(3)调用类中的静态成员时 //静态加载
(4)通过反射 //动态加载
3 类加载过程图
4 类加载各阶段完成的任务
(1)加载阶段
JVM在该阶段的主要目的是将字节码从不同的数据源(可能时class文件、也可能是jar包,甚至网络)转化成二进制字节流加载到内存中,并生成一个代表该类的java.lang.Class对象。
(2)连接阶段 - 验证
① 目的是为了确保 Class 文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身的安全;
② 包括:文件格式验证(是否以魔数 oxcafebabe 开头)、元数据验证、字节码验证和符号引用验证;
③ 可以考虑使用 -Xverify:none 参数来关闭大部分的类验证措施,缩短虚拟机类加载的时间。
(3)连接阶段 - 准备
① JVM会在该阶段堆静态变量,分配内存并初始化(对应数据类型的默认初始值,如0,0L,null,false等)。这些变量所使用的内存会在方法区中进行分配。
举例:
public class ClassLoad02 {
public static void main(String[] args) {
}
}
class A {
//属性-成员变量-字段
//1. n1 是实例属性, 不是静态变量,因此在准备阶段,是不会分配内存
//2. n2 是静态变量,分配内存 n2 是默认初始化 0 ,而不是20
//3. n3 是static final 是常量, 他和静态变量不一样, 因为一旦赋值就不变 n3 = 30
public int n1 = 10;
public static int n2 = 20;
public static final int n3 = 30;
}
(4)连接阶段 - 解析
① 虚拟机将常量池内的符号引用替换成直接引用的过程。
(5)Initialization(初始化)
① 到初始化阶段,才真正开始执行类中定义的Java程序代码,此阶段是执行()方法的过程;
② ()方法是由编译器按语句在源文件中出现的顺序,一次自动收集类中的所有静态变量的赋值动作和静态代码块中的语句,并进行合并;
③ 虚拟机会保证一个类的()方法在多线程环境中被正确的枷锁、同步,如果多个线程同时区初始化一个类,那么只会有一个线程去执行这个类的()方法,其他线程都需要阻塞等待,知道活动线程执行()方法完毕。
/**
* 演示类加载-初始化阶段
*/
public class ClassLoad03 {
public static void main(String[] args) throws ClassNotFoundException {
//1. 加载B类,并生成 B的class对象
//2. 链接 num = 0
//3. 初始化阶段
// 依次自动收集类中的所有静态变量的赋值动作和静态代码块中的语句,并合并
/*
clinit() {
System.out.println("B 静态代码块被执行");
//num = 300;
num = 100;
}
合并: num = 100
*/
//new B();//类加载
//System.out.println(B.num);//100, 如果直接使用类的静态属性,也会导致类的加载
//看看加载类的时候,是有同步机制控制
/*
protected Class<?> loadClass(String name, boolean resolve)
throws ClassNotFoundException
{
//正因为有这个机制,才能保证某个类在内存中, 只有一份Class对象
synchronized (getClassLoadingLock(name)) {
//....
}
}
*/
B b = new B();
}
}
class B {
static {
System.out.println("B 静态代码块被执行");
num = 300;
}
static int num = 100;
public B() {//构造器
System.out.println("B() 构造器被执行");
}
}
num = 300;
}
static int num = 100;
public B() {//构造器
System.out.println("B() 构造器被执行");
}
}