在虚拟机规范中,有5种情况下必须立即对类进行“初始化”
1:遇到new,getstatic,putstatic,invokestatic这4条字节码指令时,如果类没有进行过初始化,则需要先触发其初始化。
2:使用java.lang.reflect包的方法对类进行反射调用的时候,如果类没有进行过初始化,则需要先触发其初始化。
3:当初始化一个类的时候,如果发现其父类还没有进行初始化,则需要先触发其父的初始化。
4:当虚拟机启动时,用户需要指定一个要执行的主类(包含main()方法的那个类),虚拟机会先初始化这个主类。
5:当使用JDK1.7的动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后解析结果REF_getStatic、REF_putStatic、REF_invokeStatic的方法句柄,并且这个方法句柄对应的类没有进行过初始化,则需要先出发其初始化。
对于这5中会触发类进行初始化的场景,除此之外,所有引用类的方式都不会触发初始化,称为被动引用。
下面简单介绍比较常见的3种被动引用:
1:通过子类引用父类的静态字段,不会导致子类初始化:
import org.springframework.classify.SubclassClassifier;
/**
* Created by 赵海天 on 2017/8/21.
*/
class SuperClass {
static {
System.out.println("SuperClass init!");
}
public static int value = 123;
}
class SubClass extends SuperClass {
static {
System.out.println("SubClass init!");
}
}
public class NotInitialization {
public static void main(String[] args) {
System.out.println(SubClass.value);
}
}
发现运行结果只输出“SuperClass init!”,而不会输出“SubClass init!”
2:通过数组定义来引用类,不会触发此类的初始化:
import org.springframework.classify.SubclassClassifier;
/**
* Created by 赵海天 on 2017/8/21.
*/
class SuperClass {
static {
System.out.println("SuperClass init!");
}
public static int value = 123;
}
class SubClass extends SuperClass {
static {
System.out.println("SubClass init!");
}
}
public class NotInitialization {
public static void main(String[] args) {
SuperClass[] sca = new SuperClass[10];
}
}
运行结果发现没有输出“SuperClass init!”
3:常量在编译阶段会存入调用类的常量池中,本质上并没有直接引用到定义常量的类,因此不会触发定义常量的类的初始化:
import org.springframework.classify.SubclassClassifier;
/**
* Created by 赵海天 on 2017/8/21.
*/
class ConstClass {
static {
System.out.println("ConstClass init!");
}
public static final String HELLOWORLD = "hello world";
}
public class NotInitialization {
public static void main(String[] args) {
System.out.println(ConstClass.HELLOWORLD);
}
}
发现运行结果没有输出“ConstClass init!”
这里我们主要介绍了类的加载过程,接口的加载过程和类的加载过程有些不同,具体的差别在接口的内容中再讨论吧