类从被加载到虚拟机内存开始,到卸载出内存为止,它的整个生命周期包括如下 7 个阶段:java
验证、准备、解析 3 个阶段统称为链接。数组
加载、验证、准备、初始化和卸载这 5 个阶段的顺序是肯定的,类的加载过程必须按照这种顺序循序渐进地开始(注意是“开始”,而不是“进行”或“完成”),而解析阶段则不必定:它在某些状况下能够在初始化后再开始,这是为了支持 Java 语言的运行时绑定。spa
Java 虚拟机规范没有强制约束类加载过程的第一阶段(即:加载)何时开始,但对于“初始化”阶段,有着严格的规定。有且仅有 5 种状况必须当即对类进行“初始化”:code
这 5 种场景中的行为称为对一个类进行主动引用,除此以外,其它全部引用类的方式都不会触发初始化,称为被动引用。blog
/** * 被动引用 Demo1: * 经过子类引用父类的静态字段,不会致使子类初始化。 * * @author ylb * */ class SuperClass { static { System.out.println("SuperClass init!"); } public static int value = 123; } class SubClass extends SuperClass { static { System.out.println("SubClass init!"); } } public class NotInitialization { public static void main(String[] args) { System.out.println(SubClass.value); // SuperClass init! } }
对于静态字段,只有直接定义这个字段的类才会被初始化,所以经过其子类来引用父类中定义的静态字段,只会触发父类的初始化而不会触发子类的初始化。继承
/** * 被动引用 Demo2: * 经过数组定义来引用类,不会触发此类的初始化。 * * @author ylb * */ public class NotInitialization { public static void main(String[] args) { SuperClass[] superClasses = new SuperClass[10]; } }
这段代码不会触发父类的初始化,但会触发“[L 全类名”这个类的初始化,它由虚拟机自动生成,直接继承自 java.lang.Object,建立动做由字节码指令 newarray 触发。接口
/** * 被动引用 Demo3: * 常量在编译阶段会存入调用类的常量池中,本质上并无直接引用到定义常量的类,所以不会触发定义常量的类的初始化。 * * @author ylb * */ class ConstClass { static { System.out.println("ConstClass init!"); } public static final String HELLO_BINGO = "Hello Bingo"; } public class NotInitialization { public static void main(String[] args) { System.out.println(ConstClass.HELLO_BINGO); } }
编译经过以后,常量存储到 NotInitialization 类的常量池中,NotInitialization 的 Class 文件中并无 ConstClass 类的符号引用入口,这两个类在编译成 Class 以后就没有任何联系了。生命周期
接口加载过程与类加载过程稍有不一样。内存
当一个类在初始化时,要求其父类所有都已经初始化过了,可是一个接口在初始化时,并不要求其父接口所有都完成了初始化,当真正用到父接口的时候才会初始化。rem