Class.forName() 与 xxx.class的区别
所有的类都是在对其第一次使用时,动态加载到 JVM。当程序创建第一个对类的静态成员的引用时,就会加载这个类。这也证明了构造器也是类的静态方法,即使在构造器之前并没有 static 关键字。因此,使用 new 操作符创建类的对象也会被当作对类的静态成员的引用。
因此 Java 程序在他开始运行之前并非被完全加载,其各个部分是在需要时才加载的。
类加载器首先槛车这个类的 Class 对象是否已经加载。如果尚未加载,默认的类加载器就会根据类名称查找 .class 文件。在这个类的字节码被加载时,它们会接收验证,以确保其没有被破坏。
Class 对象尽在需要的时候才被加载。static 初始化是在类加载时进行的。
对于 Class.forName(),这个方法是 Class 类的一个 static 成员。Class 对象就和其他对象一样,我们可以获取并操作他的引用。forName() 是取得 Class 对象引用的一种方法。他是用一个包含目标类的文本名的 String 作为输入参数,返回一个 Class 对象的引用。forName() 方法发现某个累还没被加载,就会主动去加载这个类,在加载过程中,该类的是 static 子句将会被执行。
Java 还提供了另一种方法来生成对 Class 对象的引用,即使用类字面常量。形如:xxx.class 的形式。但是这种形式创建 Class 对象引用时,不会自动初始化 Class 对象。初始化被延迟到了对静态方法或者非常数静态域首次引用时才执行。
public class Initable { static final int staticFinal = 47; static final int staticFianl2 = ClassInitialization.rand.nextInt(1000); static { System.out.println("Initializing Initable"); } }
public class Initable2 { static int staticNonFinal = 147; static { System.out.println("Initializing Initable2"); } }
public class Initable3 { 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 = Initable.class; System.out.println("after creating Initable ref"); //Does not trigger initialization System.out.println(Initable.staticFinal); //Does trigger initialization System.out.println(Initable.staticFianl2); //Does trigger initialization System.out.println(Initable2.staticNonFinal); Class initable3 = Class.forName("chapter13.t2.Initable3"); System.out.println("after creating Initable3 ref"); System.out.println(Initable3.staticNonFinal); } }
执行输出结果:
after creating Initable ref 47 Initializing Initable 258 Initializing Initable2 147 Initializing Initable3 after creating Initable3 ref 74
从对 initable 引用的创建中可以看到,仅使用 .class 语法来获取对类的引用不会发生初始化。但是 Class.forName() 立即就进行了初始化。如果一个 static final 值是“编译期常量”,就像 Initable.staticFinal 那样,那么这个值不需要对 Initable 类进行初始化就可以被读取。但是如果只是将一个域设置为 static 和 final 的,还不足以确保这种行为,例如 Initable.staticFianl2 的访问将强制进行类型的初始化,因为他不是一个编译器常量。