当使用“.class”来创建对Class对像的引用时,不会自动地初始化该Class对像。为了使用类而做的准备工作实际包含三个步骤:
1、加载,这是由类加载器执行的。该步骤将查找字节码(通常在classpath所指定的路径中查找,但这并非是必需的),并从这些字节码中创建一个Class对像。
2、链接。在链接阶段将验证类中的字节码,为静态城分配存储空间,并且如果必需的话,将解析这个类创建的对其它类的所有引用。
3、初始化。如果该类具有超类,则对其初始化,执行静态初始化器和静态初始化块。
初始化被延迟到了对静态方法(构造器隐式的是静态的)或者非常数静态域进行首次引用时才执行
例:
class ShapeDemo4 {
static final int staticFinal = 47;
static final int staticFinal2 = ClassInitialization.rand.nextInt(1000);
static {
System.out.println("Initializing Initable");
}
public class Initbale2 {
static int staticNonFinal=147;
static{
System.out.println("Initializing Initable2");
}
}
public class Initbale3 {
static int staticNonFinal=74;
static{
System.out.println("Initializing Initable3");
}
}
public class ClassInitialization {
public static Random rand=new Random(47);
public static void main(String[] args) throws ClassNotFoundException {
Class initable=ShapeDemo3.class;
System.out.println("AFTER");
System.out.println(ShapeDemo4.staticFinal);
System.out.println(ShapeDemo4.staticFinal2);
System.out.println(Initbale2.staticNonFinal);
Class initaClass=Class.forName("com.shape.Initbale3");
System.out.println("After");
System.out.println(Initbale3.staticNonFinal);
}
}
AFTER
47
Initializing Initable
258
Initializing Initable2
147
Initializing Initable3
After
74
从上面程序运行的结果可以看出初始化尽可能的实现“惰性”。从对SheapDemo4引用的创建中可以看到,当只使用.class语法来获得对类的引用时不会引发初始化。但是,为了产生Class引用,Class.forName()立即就进行了初始化。
如果一个static final值是“编译期常量”,那么这个值不需要对类进行初始化就可以被读取,如果例子中的“static final int staticFinal = 47;”但是如果只是将一个域设置为static或final时,还不足以确保这种行为,例如对“static int staticNonFinal=147;”进行访问时将强制进行初始化,因为它不是一个编译期常量。
如果一个static域不是final的,那么在对它访问时,总是要求在它被读取之前,要先进行链接(即为这个域分配存储空间)和初始化(初始化该在存储空间),例如对Initable.staticNonFinal的访问中所看到的结果。
第一次博客,基本按书上意思转达。因自己写作水平有限,请朋友们多多指教,以后会抽更多的时间记录。