包含三个步骤:
一、加载:这是由类加载器执行的。该步骤将查找字节码(通常在classpath所指定的路径中查找,但这并非是必需的),并从这些字节码中创建一个Class对象。
二、链接:在链接阶段将验证类中的字节码,为静态域分配存储空间,并且如果必需的话,将解析这个类创建的对其他类所有引用。
三、初始化:如果该类具有超类,则对其初始化,执行静态初始化器和静态初始化块。
初始化被延迟到了对静态方法(构造器隐式地是静态的)或者非常数静态域进行首次引用时才执行。
package classTest;
public class ClassInitialization {
public static void main(String[] args) throws ClassNotFoundException {
Class c = Initable.class;
// Initable initable = new Initable();
System.out.println("begin Initable");
System.out.println(Initable.staticInt);
System.out.println(Initable.staticInt2);
System.out.println("-------------------------");
System.out.println("begin Initable2222222222222222");
System.out.println(Initable2.staticInt);
System.out.println("-------------------------");
Class c3 = Class.forName("classTest.Initable3");
System.out.println("begin Initable33333333333333");
System.out.println(Initable3.staticInt);
}
}
class Initable {
static final int staticInt = 1;
static final int staticInt2 = 2;
static {
System.out.println("Initable");
}
}
class Initable2 {
static int staticInt = 22;
static {
System.out.println("Initable2");
}
}
class Initable3 {
static final int staticInt = 33;
static {
System.out.println("Initable3");
}
}
/* OutPut:
begin Initable
1
2
-------------------------
begin Initable2222222222222222
Initable2
22
-------------------------
Initable3
begin Initable33333333333333
33
*///
从对Initable引用的创建中可以看出,使用.class语法来获取对类的引用不会引发初始化。但是Initable3,为了产生Class引用,Class.forName()立即就进行了初始化。
如果一个static final值是“编译期常量”,就像Initable.staticInt,那么这个值不需要对Initable类进行初始化就可以被读取。
如果一个static域不是final的,那么在对它访问时,总是要求在它被读取之前,要先进行链接(为这个域分配存储空间)和初始化(初始化该存储空间),就像访问Initable2.staticInt。
——《Java编程思想》