全面解析JVM加载中初始化的时机

JVM类加载过程

JVM类加载过程分为几个阶段,分别是加载验证准备解析初始化加载是把二进制字节码载入内存,验证是校验字节流中包含的信息是否符合当要求,准备是为静态变量分配内存并设置静态变量初始值,解析是把常量池内的符号引用替换为直接引用,初始化是执行全部静态变量的赋值动做和静态语句块中的语句。更多详尽分析请阅读以前的文章《JVM的类加载机制全面解析》,这里再也不赘述了。java

欢迎关注微信公众号:万猫学社,每周一分享Java技术干货。数组

类初始化的时机

对于咱们开发人员,我认为应该具体了解一下初始化阶段何时在开始。JVM规范对此作了严格规范,有且只有如下5种状况必须对类进行初始化:微信

  1. 遇到new、getstatic、putstatic或invokestatic这四条字节码指令时,若是类没有被初始化过,就须要先进行初始化。对于字节码指令不了解的同窗,可能就是一脸蒙圈了。咱们来讲人话,就是:使用new关键字实例化对象的时候、读取和设置一个类的静态字段(不被final修饰的)和调用一个类的静态方法的时候。这样说更容易被理解一些。jvm

  2. 使用java.lang.reflect包中的方法对类进行反射调用的时候,若是类没有被初始化过,就须要先进行初始化。spa

  3. 当初始化一个类的时候,若是发现它的父类尚未被初始化过,就须要先初始化它的父类。rest

  4. JVM会先初始化要执行的主类,也是包含main()方法的那个类。code

  5. 当使用JDK 1.7的动态语言支持时,若是java.lang.invoke.MethodHandle实例最后的解析结果是REF_getStatic(使用MethodHandle读取类的静态字段)、REF_putStatic(使用MethodHandle设置类的静态字段)、REF_invokeStatic(使用MethodHandle调用类的静态方法)的方法句柄时,若是这个方法句柄没有被初始化过,就须要先进行初始化。对象

欢迎关注微信公众号:万猫学社,每周一分享Java技术干货。继承

被动引用

刚刚提到的5种状况,都会触发初始化,这些行为为称为对一个类的主动引用。除了这些之外,全部引用类的方式都不会触发初始化,被为被动引用。为了更好的理解,下面举几个被动引用的例子。接口

经过子类引用父类的静态变量

public class SuperClass {
    static {
        System.out.println("父类正在初始化");
    }

    public static String name = "万猫学社";
}
public class SubClass extends SuperClass {
    static {
        System.out.println("子类正在初始化");
    }
}
public class OneMoreStudy {
    public static void main(String[] args) {
        System.out.println(SubClass.name);
    }
}

对于静态变量,只有直接定义这个变量的类才会被初始化,经过子类引用父类中定义的静态变量,只会触发父类的初始化而不会触发子类的初始化,运行的结果是:

父类正在初始化
万猫学社

结果中并无“子类正在初始化”。

欢迎关注微信公众号:万猫学社,每周一分享Java技术干货。

经过数组定义来引用类

public class OneMoreStudy {
    public static void main(String[] args) {
        SuperClass[] arrays = new SuperClass[10];
        System.out.println("数组元素个数:" + arrays.length);
    }
}

这段代码中使用以前的SuperClass类,定义了一个SuperClass类的一维数组,运行后的结果是:

数组元素个数:10

结果中并无“父类正在初始化”,说明并无触发SuperClass类的初始化。实际上,有一个名为“[LSuperClass”的类被初始化了,它是由JVM自动生成的、直接继承于java.lang.Object,建立动做由字节码指令newarray触发。

欢迎关注微信公众号:万猫学社,每周一分享Java技术干货。

常量

public class ConstClass {
    static {
        System.out.println("有常量的类正在初始化");
    }

    public static final String NAME = "万猫学社";
}
public class OneMoreStudy {
    public static void main(String[] args) {
        System.out.println(ConstClass.NAME);
    }
}

常量在编译阶段会存入调用类的常量池中,本质没有直接引用到定义的常量的类,不会触发定义常量的类的初始化,因此运行的结果是:

万猫学社

结果中并无“有常量的类正在初始化”。

欢迎关注微信公众号:万猫学社,每周一分享Java技术干货。

接口初始化的时机

接口也有初始化过程,和类是一致的。不过接口中不能使用“static{}”语句块,但编译器仍然会为接口生成“clinit()”类构造器,用于初始化接口中所定义的成员变量。

接口初始化的时机,基本和以前提到的类的5种状况基本一致,惟一不同的是第3种状况:在一个类被初始化时,它的父类也必须被初始化,可是一个接口被初始化时,它的父接口并不要求被初始化。只有在真正使用到父接口时才会被初始化,好比:引用父接口中定义的常量。

欢迎关注微信公众号:万猫学社,每周一分享Java技术干货。

结语

此次主要分享了类在何时被初始化,共有5种状况。除了这种5种状况的引用叫作被动引用,同时举了3个被动引用的例子。同时,也提到初始化接口和类有什么不一样。

欢迎关注微信公众号:万猫学社,每周一分享Java技术干货。

相关文章
相关标签/搜索