JVM -- 类的初始化

时间:2023-05-09 15:06:49

《深入理解Java虚拟机》 第二版中介绍到了类的加载过程。

一个类从加载入内存到卸载出内存为止,整个生命周期包括:

Loading(加载)-----Verification(验证)-----Preparation(准备)-----Resolution(解析)-----Initialization(初始化)-----Using(使用)----Unloading(卸载)

在Initalization 阶段,虚拟机规范严格定一个5种必须立即对类初始化的情况。

1、遇到new ,getstatic putstatic invokestatic

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

3、初始化一个类的时候,如果发现其父类还未初始化,那么首先初始化其父类。

4、当虚拟机启动的时候,用户需要指定一个需要执行的主类,然后虚拟机才会初始化这个主类。

5、JDK1.7的动态语言支持,如果一个java.lang.invoke.MethodHandle 实例最后解析结果REF_getStatic,REF_putStatic,REF_invokeStatic 的方法句柄,并且这个方法句柄对应的类没有进行初始化,则需要将其初始化。

除此之外,所有引用类的方式都不会触发初始化。---被动引用。

1、通过子类调用父类中定义的静态字段,不会导致子类的初始化

 package volshell;

 public class SuperCalss {
static {
System.out.println("super class init");
}
public static int value = ;
} package volshell; public class SubClass extends SuperCalss {
static {
System.out.println("subclass init");
}
} package volshell; public class Demo1 {
public static void main(String[] args) {
System.out.println(SubClass.value);
}
}

输出:

  super class init
  23
对于静态字段,只有直接定义该字段的类才会被初始化,因此通过子类的来引用父类定义的静态字段,并不会触发子类的初始化。

2、使用数组来定义引用类,不会触发该类的初始化。

  还是使用上面的代码在main方法中添加

  SuperCalss[]subClass = new SuperCalss[10];

执行之后不会有任何的输出。

3、常量在编译接阶段就会存入调用类的常量池中,本质上没有直接引用定义常量的类,因此不会触发被引用的类的初始化。

  同样还是使用上面的例子 ,将superClass 中的 value 设置为public static final int value=23; 设置为常量

在主类中调用SuperClass.value

执行结果;

  23

在编译阶段,通过常量的传播优化,已将常量value存入Demo1 的常量池中了。