基本说明:
反射机制是java实现的动态语言的关键,也就是通过反射实现类动态加载。
- 静态加载:编译时加载相关的类,如果没有则报错,依赖性太强。
- 动态加载:运行时加载需要的类。如果运行时不用该类,即使没有也不报错,降低了依赖性。
类加载的时机:
- 创建对象时。 静态加载
- 当子类被加载,父类也加载 静态加载
- 调用类中的静态成员 静态加载
- 通过反射 动态加载
类加载的三个阶段:
概述
- 加载 Loading:将class文件读入内存,并创建一个java.lang.Class对象。此类过程由类的加载器完成。
- 连接Linking:验证阶段 verification 准备阶段 preparation 解析 resolution
将类的二进制数据合并到JRE中 - 初始化 initialization JVM负责对类进行初始化,主要是静态成员
加载 Loading:
JVM在该阶段的主要目的是将字节码从不同的数据源(可能是Class文件,也可能是Jar包,甚至是网络)转化为二进制字节流加载到内存,并生成 java.lang.Class对象。
连接阶段–验证阶段
- 目的是为了保证Class文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机的自身的安全。
- 文件格式的验证(是否以魔数0xcafebabe开头) 、元数据验证、字节码验证和符号引用验证。
- 可以考虑使用 - Xverify:none 参数来关闭大部分的类验证措施,缩短虚拟机类加载时间。
连接阶段-准备
- JVM会在该阶段对静态变量,分配内存并进行默认初始化(对应数据类型的默认初始值,如 0、0L、null、false等)。这些变量所使用的内存都将在方法区中进行分配。
连接阶段-解析
- 虚拟机将常量池内的符号引用替换为直接引用。
Initialization(初始化)
- 到了初始化的阶段,才真正意义上开始执行类中定义的Java程序代码,此阶段是执行<clinit>()方法的过程。
- <clinit>()方法是由编译器按照语句在源文件中出现的顺序,依次自动收集中的所有静态变量的赋值操作和静态代码块中的语句,并进行合并
- 虚拟机会保证一个类的<clinit>()方法在多线程环境中被正确的加锁、同步,如果多线程同时去初始化一个类,那么只有一个线程去执行这个类的<clinit>()方法,其他线程等待。