一篇文章带你读懂反射(三)

120 阅读3分钟

开启掘金成长之旅!这是我参与「掘金日新计划 · 12 月更文挑战」的8天,点击查看活动详情

三.类加载

1.动态加载和静态加载

  • 基本说明

    反射机制是java实现动态语言的关键,也就是通过反射实现类动态加载 1.静态加载:编译时加载相关的类,如果没有则报错,依赖性太强 2.动态加载:运行时加载需要的类,如果运行时不用该类,即使不存在该类,则不报错,降低了依赖性 3.举例说明

  • 类加载时机

    1.当创建对象时(new))

    2.当子类被加载时

    3.调用类中的静态成员时

    4.通过反射 Class.forName("com.test.Cat");

2.类加载流程图

image-20221121170527762

image-20221121170012191

  • 类加载各阶段完成任务 image-20221121170256381

3.类加载的五个阶段

3.1加载阶段

  • JVM在该阶段的主要目的是将字节码从不同的数据源(可能是class文件、也可能是jar包,甚至网络)转化为二进制字节流加载到内存中,并生成一个代表该类的java.lang.Class对象

3.2连接阶段

3.2.1验证

1.目的是为了确保Class文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身的安全。

2.包括:文件格式验证(是否以魔数oxcafebabe开头)、元数据验证、字节码验证和符号引用验证

3.可以考虑使用-Xverify:none参数来关闭大部分的类验证措施,缩短虚拟机类加载的时间。

3.2.2准备

JVM会在该阶段对静态变量,分配内存并初始化(对应数据类型的默认初始值如0、0L、null、false等)。这些变量所使用的内存都将在方法区中进行分配

/**
 * @author LeeZhi
 * @version 1.0
 * 我们说明一个类加载的连接阶段-准备
 */
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;
}
3.2.3解析

虚拟机将常量池内的符号引用替换为直接引用的过程。

3.3初始化

  • Initialization(初始化)

1.到初始化阶段,才真正开始执行类中定义的Java程序代码,此阶段是执行<clinit>()方法的过程。

2.<clinit>()方法是由编译器按语句在源文件中出现的顺序,依次自动收集类中的所有静态变量的赋值动作和静态代码块中的语句,并进行合并。

3.虚拟机会保证一个类的()方法在多线程环境中被正确地加锁、同步,如果多个线程同时去初始化一个类,那么只会有一个线程去执行这个类的()方法,其他线程都需要阻塞等待,直到活动饯程执行<clinit>()方法完毕[debug源码]

```
/**
 * @author LeeZhi
 * @version 1.0
 * 演示类加载-初始化阶段
 */
public class ClassLoad03 {
    public static void main(String[] args) {
        //1 加载B类,并生成B的class对象
        //2. 链接num=0
        //3. 初始化阶段
        //依次自动收集类中的所有静态变量的赋值动作和静态代码块中的语句
        /*
            clinit(){
                System.out.println("B静态代码块被执行");
                //num=300:
                num=100;
            }
            合并:num = 100
         */
        //4. "B () 构造器被执行"
        //new B();//类加载
        System.out.println(B.num);//100  如果直接使用类的静态属性,也会导致类的加载
    }
}
class B{
    static {
        System.out.println("B 静态代码块被执行");
        num=300;
    }
    static int num = 100;
    public B(){
        System.out.println("B () 构造器被执行");
    }
}
```