java编程思想第四版 14.2.1
当使用 “.class” 来创建对Class对象的引用时,不会自动地初始化该Class对象。为了使用类而做的准备工作实际包括三个步骤:
1.加载,这是由类加载器执行的。该步骤将查找字节码(非必须),并从这些字节码中创建一个Class对象。
2.链接。在链接阶段将验证类中的字节码,为静态域分配存储空间,并且如果必需的话,将解析这个类创建的对其他类的所有引用。
3.初始化。如果该类具有超类,则对其初始化,执行静态初始化器和静态初始化块。
初始化被延迟到了对静态方法(构造器隐式地是静态的)或者非常数静态域进行首次引用时才执行:
package com.a;
import java.util.Random;
public class InitTest {
public static Random rand = new Random(47);
public static void main(String[] args) throws ClassNotFoundException {
Class<?> i1 = Initable1.class;
System.out.println("After create Initable1 ref");
// Does not trigger initialization
System.out.println(Initable1.staticFinal1);
// Does trigger initialization
System.out.println(Initable1.staticFinal2);
// Does trigger initialization
System.out.println(Initable2.staticNonFinal);
Class<?> i3 = Class.forName("com.a.Initable3");
System.out.println("After create Initable3 ref");
System.out.println(Initable3.staticNonFinal);
}
}
class Initable1 {
static final int staticFinal1 = 47;
static final int staticFinal2 = InitTest.rand.nextInt(1000);
static {
System.out.println("Initializing Initable1");
}
}
class Initable2 {
static final int staticNonFinal = 147;
static {
System.out.println("Initializing Initable2");
}
}
class Initable3 {
static final int staticNonFinal = 74;
static {
System.out.println("Initializing Initable3");
}
}
After create Initable1 ref
47
Initializing Initable1
258
Initializing Initable2
147
Initializing Initable3
After create Initable3 ref
74
初始化有效的实现了尽可能的“惰性”。从对Initable1引用的创建中可以看到,仅使用.class语法来获取对类的引用不会引发初始化。但是,为了产生Class引用,Class.forName()立即就进行了初始化,就像在对Initable3引用的创建中所看到的。
如果一个static final 值是“编译期常量”,就像Initable.staticFinal1那样,那么这个值不需要对Initable类进行初始化就可以被读取。但是,如果只是将一个域设置为static final的,还不足以确保这种行为,例如,对Initable.staticFinal2的访问将强制进行类的初始化,因为它不是一个编译期常量。
如果一个static域不是final的,那么在对它访问时,总是要求在它被读取之前,要先进行链接(为这个域分配存储空间)和初始化(初始化该存储空间),就像在对Initable2.staticNonFinal的访问中所看到的那样。