Java哪些情况下必须对类进行“初始化”

  • 遇到new、getstatic、putstatic或invokestatic这四条字节码指令时,如果类没有进行初始化,则需要先触发其初始化。生成这四条指令的常见场景是:
  • 使用new关键字实例化对象的时候
  • 读取或设置一个类的静态字段的时候(被final修饰、已在编译期把结果放入常量池的静态字段除外)
  • 调用一个类的静态方法的时候
  • 使用java.lang.reflect包的方法对类进行反射调用的时候,如果类没有进行初始化,则需要先触发其初始化
  • 当初始化一个类的时候,如果发现其父类还没有进行过初始化,则需要先触发其初始化
  • 当虚拟机启动时,用户需要指定一个要执行的主类(包含main方法的那个类),虚拟机会先初始化这个主类
  • 当使用JDK1.7的动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后的解析结果REF_getStatic、REF_putStatic、REF_invokeStatic的方法句柄,并且这个方法句柄所对应的类没有进行初始化,则需要先触发其初始化

注意有且只有上面这五种情况下才会触发类的初始化

这些情况下不会对类进行初始化:

  • 通过子类引用父类的静态字段,不会导致子类的初始化
  • 通过数组定义来引用类,不会触发此类的初始化
  • 常量在编译阶段会存入调用类的常量池中,本质上没有直接引用到定义常量的类,因此不会触发定义常量的类的初始化

实例如下:

public class SuperClass {
    static {
        System.out.println("SuperClass init!!");
    }
    public static int value = 123;
}

public class SubClass extends SuperClass {
    static {
        System.out.println("SubClass init!!");
    }
}

public class ConstClass {
    static {
        System.out.println("ConstClass init!!");
    }
    public static final String HELLOWORLD = "hello world!";
}
// 测试
public class NotInitialization {
    public static void main(String args[]) {

        /**
         * 通过子类引用父类的静态字段,不会导致子类的初始化
         * 打印结果:
         *   SuperClass init!!
         *   123
         */
//        System.out.println(SubClass.value);

        /**
         * 通过数组定义来引用类,不会触发此类的初始化
         * 打印结果:
         *   无
         */
//        SuperClass[] sca = new SuperClass[10];

        /**
         * 常量在编译阶段会存入调用类的常量池中,本质上没有直接引用到定义常量的类,因此不会触发定义常量的类的初始化
         * 打印结果:
         *   hello world!
         */
        System.out.println(ConstClass.HELLOWORLD);

    }
}