java 类初始化_Java类初始化的时机详解

1. 概述

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

3afa5d24bf71

类的生命周期

加载、验证、准备、初始化和卸载这5个阶段的顺序是确定的,类的加载过程

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

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

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

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

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

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

当使用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 clazz = SubClass.class;

Constructor 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虚拟机》

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值