当Java程序需要使用某个类时,如果该类还未被加载到内存中,JVM会通过加载、连接(验证、准备和解析)、初始化三个步骤来对该类进行初始化。
java类的生命周期:
类加载包括加载、连接和初始化三个过程。
加载:
这里的class文件泛指各种来源的二进制文件,可以是本地文件,也可以是来自网络等。
如果输入数据不是 ClassFile 的结构,则会抛出 ClassFormatError。加载阶段是用户参与的阶段,我们可以自定义类加载器,去实现自己的类加载过程。
连接:
连接是把原始的类定义信息平滑地转入 JVM 运行的过程。
这里可进一步细分成验证、准备和解析三个步骤:
- 验证(Verification)是虚拟机安全的重要保障,JVM 需要对class文件的格式、元数据、字节码信息进行验证,保证符合 Java 虚拟机规范的,防止恶意信息或者不合规信息危害 JVM 的运行。
-
准备(Pereparation),创建类或者接口中的静态变量,并初始化静态变量的初始值。
-
解析(Resolution),在这一步会将常量池中的符号引用(symbolic reference)替换为直接引用。
假如 A.class文件被编译成Class类A,且在A中引用了一个具体的实现类B,,在编译阶段A不知B是否已经被编译,为了能够正确找到B,在.class文件中将使用一个字符串来表示B的地址,这就被称为‘符号引用’。在运行时,当A发生了类加载,就会在本阶段发现B未加载,那将会触发B的类加载,将B加载到JVM中,此时A中对B的符号引用就会被替换为直接引用,该过程也被称为静态解析。如果A引用的B是一个抽象类或者接口,它有C和D两种实现类,那么当触发B的类加载时,也无法确定B的具体实现时哪个,因此无法将A中对B的符号引用替换为直接引用,此时只能等到运行过程中发生了调用,此时JVM调用栈中将会得到B的具体实现类的信息,这时候再进行解析操作,将A中对B的符号引用替换为直接引用,这个过程被称为’动态解析‘。
初始化:
初始化阶段(initialization),这一步真正去执行类初始化的代码逻辑,包括静态字段赋值的动作,以及执行类定义中的静态初始化块内的逻辑,编译器在编译阶段就会把这部分逻辑整理好,父类型的初始化逻辑优先于当前类型的逻辑。