Java类的初始化
初始化阶段是执行类构造器clinit方法的过程,clinit方法由类变量的赋值动作和静态语句块按照在源文件出现的顺序合并而成,该合并操作由编译器完成。
- 方法clinit对于类或接口不是必须的,如果一个类中没有静态代码块,也没有静态变量的赋值操作,那么编译器不会生成
- 方法clinit方法与实例构造器不同,不需要显式的调用父类的
- 为了防止多次执行clinit,虚拟机会确保clinit方法在多线程环境下被正确的加锁同步执行,如果有多个线程同时初始化一个类,那么只有一个线程能够执行clinit方法,其它线程进行阻塞等待,直到clinit执行完成。
- 注意:执行接口的clinit方法不需要先执行父接口的clinit,只有使用父接口中定义的变量时,才会执行。
- 初始化过程的主要操作是执行静态代码块和初始化静态域。
- 在一个类被初始化之前,它的直接父类也需要被初始化。
类初始化场景
虚拟机中严格规定了有且只有5种情况必须对类进行初始化。
- 执行new、getstatic、putstatic和invokestatic指令;
- 使用reflect对类进行反射调用;
- 初始化一个类的时候,父类还没有初始化,会事先初始化父类;
- 启动虚拟机时,需要初始化包含main方法的类;
- 在JDK1.7中,如果java.lang.invoke.MethodHandler实例最后的解析结果REF_getStatic、REF_putStatic、REF_invokeStatic的方法句柄,并且这个方法句柄对应的类没有进行初始化;
不会触发类初始化的情况
- 通过子类引用父类的静态字段,只会触发父类的初始化,而不会触发子类的初始化。
class Parent {
static int a = 100;
static {
System.out.println("parent init!");
}
}
class Child extends Parent {
static {
System.out.println("child init!");
}
}
public class Init{
public static void main(String[] args){
System.out.println(Child.a); //不会初始化类Child
}
}
输出结果为:parent init! 不会初始化Child类。
- 定义对象数组,不会触发该类的初始化。
public class Init{
public static void main(String[] args){
Parent[] parents = new Parent[10]; //不会初始化类Parent
}
}
- 常量在编译期间会存入调用类的常量池中,本质上并没有直接引用定义常量的类,不会触发定义常量所在的类。
class Const {
static final int A = 100; //编译阶段,常量A存储到Init类的常量池中
static {
System.out.println("Const init");
}
}
public class Init{
public static void main(String[] args){
System.out.println(Const.A);
}
}
在编译阶段,Const类中常量A的值100存储到Init类的常量池中,这两个类在编译成class文件之后就没有联系了。
- 通过类名获取Class对象,不会触发类的初始化。
public class test {
public static void main(String[] args) throws ClassNotFoundException {
Class c_dog = Dog.class; //不会初始化Dog类
Class clazz = Class.forName("zzzzzz.Cat"); //会初始化Cat类
}
}
class Cat {
private String name;
private int age;
static {
System.out.println("Cat is load");
}
}
class Dog {
private String name;
private int age;
static {
System.out.println("Dog is load");
}
}
通过Class.forName加载指定类时,如果指定参数initialize为false时,也不会触发类初始化,其实这个参数是告诉虚拟机,是否要对类进行初始化。
通过ClassLoader默认的loadClass方法,也不会触发初始化动作;