Java类初始化的时机详解

1. 概述

类从被加载到虚拟机内存中开始,到卸载出内存为止,它的整个生命周期包括:加载(Loading)、验证(Verification)、准备(Preparation)、解析(Resolution)、初始化(Initialization)、使用(Useing)、卸载(Unloading)7个阶段。其中验证、准备和解析3个部分统称为连接(Linking),这7个阶段的发生顺序如图所示。

类的生命周期

加载、验证、准备、初始化和卸载这5个阶段的顺序是确定的,类的加载过程
必须按照这种顺序按部就班地开始,而解析阶段则不一定:它在某些情况下可以在初始化阶段之后再开始,这是为了支持Java语言的运行时绑定(也称为动态绑定或晚期绑定) 。

对于初始化阶段,虚拟机规范则是严格规定了有且只有5种情况必须立即对类进行“初始化”( 而加载、 验证、 准备自然需要在此之前开始):

  1. 遇到new、getstatic、putstatic或invokestatic这4条字节码指令时,如果类没有进行过初始化,则需要先触发其初始化。
  2. 使用java.lang.reflect包的方法对类进行反射调用的时候,如果类没有进行过初始化,则需要先触发其初始化。
  3. 当初始化一个类的时候,如果发现其父类还没有进行过初始化,则需要先触发其父类的初始化。
  4. 当虚拟机启动时,用户需要指定一个要执行的主类( 包含main()方法的那个类),虚拟机会先初始化这个主类。
  5. 当使用JDK 1.7的动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后的解析结果REF_getStatic、REF_putStatic、REF_invokeStatic的方法句柄, 并且这个方法句柄所对应的类没有进行过初始化, 则需要先触发其初始化。

这五种情况称为对一个类进行主动引用。其余情况都不会触发初始化,称为被动引用

2 主动引用

首先准备两个类用户测试其是否初始化。

SuperClass

package cn.lastwhisper.jvm.classloading.initiative;

/**
 * @author lastwhisper
 */
public class SuperClass {
    public static int value = 123;

    static {
        System.out.println("SuperClass static code init!");
    }

    public SuperClass() {
        System.out.println("SuperClass constructor init! ");
    }

    public static int getValue() {
        return value;
    }

    public static void setValue(int value) {
        SuperClass.value = value;
    }
}

SubClass 继承自SuperClass

package cn.lastwhisper.jvm.classloading.initiative;

/**
 * @author lastwhisper
 */
public class SubClass extends SuperClass {

    public static int subvalue = 456;

    static {
        System.out.println("SubClass static code init!");
    }

    public SubClass() {
        System.out.println("SubClass constructor init! ");
    }

    public static int getSubvalue() {
        return subvalue;
    }

    public static void setSubvalue(int subvalue) {
        SubClass.subvalue = subvalue;
    }
}

2.1 情景一

遇到new、getstatic、putstatic或invokestatic这4条字节码指令时,如果类没有进行过初始化,则需要先触发其初始化。

package cn.lastwhisper.jvm.classloading.initiative;

/**
 * 主动引用触发初始化、演示一
 * @author lastwhisper
 */
public class Initialization1 {
    // 遇到new、getstatic、putstatic或invokestatic这4条字节码指令时,
    // 如果类没有进行过初始化,则需要先触发其初始化。
    public static void main(String[] args) {
        // 1.new字节码指令
        //SubClass subClass = new SubClass();

        // 2.getstatic字节码指令
        // 被final修饰、已在编译期把结果放入常量池的静态字段除外
        //int subvalue = SubClass.subvalue;

        // 3.setstatic字节码指令
        // 被final修饰、已在编译期把结果放入常量池的静态字段除外
        //SubClass.subvalue = 789;

        // 4.invokestatic字节码指令
        //SubClass.getSubvalue();
    }
}

2.2 情景二

使用java.lang.reflect包的方法对类进行反射调用的时候,如果类没有进行过初始化,则需要先触发其初始化。

package cn.lastwhisper.jvm.classloading.initiative;

import java.lang.reflect.Constructor;

/**
 * 主动引用触发初始化、演示二
 * @author lastwhisper
 */
public class Initialization2 {
    public static void main(String[] args) {
        // 使用java.lang.reflect包的方法对类进行反射调用的时候,
        // 如果类没有进行过初始化,则需要先触发其初始化。

        try {
            // 使用Class.forName();也行,不要使用对象.class。
            Class<SubClass> clazz = SubClass.class;
            Constructor<SubClass> constructor = clazz.getConstructor();
            constructor.newInstance();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

2.3 情景三

当初始化一个类的时候,如果发现其父类还没有进行过初始化,则需要先触发其父类的初始化。

package cn.lastwhisper.jvm.classloading.initiative;

/**
 * 主动引用触发初始化、演示三
 * @author lastwhisper
 */
public class Initialization3 {
    // 当初始化一个类的时候,如果发现其父类还没有进行过初始化,
    // 则需要先触发其父类的初始化
    public static void main(String[] args) {
        // 在Initialization1的new指令和Initialization2的反射创建对象都有体现
        SubClass subClass = new SubClass();
        //初始化顺序
        //SuperClass static code init!  首先初始化父类静态代码块
        //SubClass static code init!    其次初始化自己的静态代码块
        //SuperClass constructor init!  其次初始化父类的构造器
        //SubClass constructor init!    其次初始化自己的构造器
    }
}

2.4 情景四

当虚拟机启动时,用户需要指定一个要执行的主类( 包含main()方法的那个类),虚拟机会先初始化这个主类。

package cn.lastwhisper.jvm.classloading.initiative;

/**
 * 主动引用触发初始化、演示四
 * @author lastwhisper
 */
public class Initialization4 {

    static {
        System.out.println("Initialization4 static code init!");
    }

    public Initialization4() {
        System.out.println("Initialization4 constructor init!");
    }

    public static void main(String[] args) {
        // 当虚拟机启动时,用户需要指定一个要执行的主类(包含main()方法的那个类),虚拟机会先初始化这个主类。
    }
}

2.5 情景五

当使用JDK 1.7的动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后的解析结果REF_getStatic、REF_putStatic、REF_invokeStatic的方法句柄, 并且这个方法句柄所对应的类没有进行过初始化, 则需要先触发其初始化。

首先创建一个MethodHandleClass类

package cn.lastwhisper.jvm.classloading.initiative;

/**
 * @author lastwhisper
 */
public class MethodHandleClass {

    static {
        System.out.println("MethodHandleClass static code init!");
    }
    public MethodHandleClass() {
        System.out.println("MethodHandleClass constructor init!");
    }

    // REF_invokeStatic
    public static void testREF_invokeStatic(String str) {
        System.out.println(str);
    }

}
package cn.lastwhisper.jvm.classloading.initiative;

import java.lang.invoke.MethodHandle;
import java.lang.invoke.MethodHandles;
import java.lang.invoke.MethodType;

/**
 * 主动引用触发初始化、演示五
 * @author lastwhisper
 */
public class Initialization5 {

    public static void main(String[] args) {

        MethodHandles.Lookup lookup = MethodHandles.lookup();
        try {
            // REF_invokeStatic
            MethodHandle testREF_invokeStatic = lookup.findStatic(MethodHandleClass.class, "testREF_invokeStatic", MethodType.methodType(void.class, String.class));
            testREF_invokeStatic.invoke("啥也不干,打印一段话");

            // REF_getStatic

            // REF_putStatic
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (Throwable throwable) {
            throwable.printStackTrace();
        }
    }
}

3. 被动引用

被动引用不会触发类的初始化

3.1 情景一

子类引用父类的静态字段,只会触发子类的加载、父类的初始化,不会导致子类初始化

package cn.lastwhisper.jvm.classloading.passive;

import cn.lastwhisper.jvm.classloading.initiative.SubClass;

/**
 * 被动使用类字段不触发初始化、演示一
 * @author lastwhisper
 */
public class NotInitialization1 {
    // 子类引用父类的静态字段,只会触发子类的加载、父类的初始化,不会导致子类初始化
    // 是否要触发子类的加载和验证,在虚拟机规范中并未明确规定,这点取决于虚拟机的具体实现
    // 对于Sun HotSpot虚拟机,可通过-XX:+TraceClassLoading参数观察到此操作会导致子类的加载
    public static void main(String[] args) {
        System.out.println(SubClass.value);
        //[Loaded cn.lastwhisper.jvm.classloading.passive.SubClass from ...]
    }
}

打印结果:子类并未初始化

SuperClass static code init!
123

3.2 情景二

通过数组定义来引用类,不会触发此类的初始化

package cn.lastwhisper.jvm.classloading.passive;

import cn.lastwhisper.jvm.classloading.initiative.SuperClass;

/**
 * 被动使用类字段不触发初始化、演示二
 * -XX:+TraceClassLoading
 * @author lastwhisper
 */
public class NotInitialization2 {
    public static void main(String[] args){
        // 通过数组定义来引用类,不会触发此类的初始化
        // 会触发L+全类名的初始化
        SuperClass[] superClasses = new SuperClass[10];
    }
}

3.3 情景三

常量在编译阶段会进行常量优化,将常量存入调用类的常量池中,
本质上并没有直接引用到定义常量的类,因此不会触发定义常量的类的初始化。

创建一个被static final修饰的类。

/**
 * @author lastwhisper
 */
public class ConstClass {
    public static final String HELLOWORLD = "hello world";

    static {
        System.out.println("ConstClass init!");
    }
}
package cn.lastwhisper.jvm.classloading.passive;

/**
 * 被动使用类字段不触发初始化、演示三
 * @author lastwhisper
 */
public class NotInitialization3 {
    public static void main(String[] args) {
        // 常量在编译阶段会进行常量优化,将常量存入**调用类**的常量池中,
        // 本质上并没有直接引用到定义常量的类,因此不会触发定义常量的类的初始化。
        System.out.println(ConstClass.HELLOWORLD);
        // hello world
    }
}

打印结果:ConstClass类并未初始化。

hello world

参考

《深入理解Java虚拟机》

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,458评论 4 363
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,454评论 1 294
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,171评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,062评论 0 207
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,440评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,661评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,906评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,609评论 0 200
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,379评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,600评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,085评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,409评论 2 254
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,072评论 3 237
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,088评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,860评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,704评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,608评论 2 270