第十四章 类型信息2.0
类字面常量
Java 还提供了另一种方法来生成Class对象的引用:使用“类字面常量(class literal)”。对上述程序来说,看起来就象下面这样:
FancyToy.class;
这样做不仅更简单,而且更安全,因为它在编译期就会受到检查。并且它无需方法调用,所以也更高效。
类字面常量不仅可以应用于普通的类,也可以应用于接口、数组以及基本数据类型。以外,对于基本数据类型的包装类,还有一个标准域TYPE。TYPE域是一个引用,指向对应的基本数据类型的Class对象,如下所示:
boolean.classBoolean.TYPE 等
我建议使用”.class”的形式,以保持它们与常规类的一致性。
注意,有一点很有趣,当使用“.class”来创建对Class对象的引用时,不会自动地初始化该Class对象。为了使用类而作的准备工作实际包括三个步骤:
1. 加载,这是由类加载器执行的。该步骤将查找字节码,并从这些字节码中创建一个Class对象。
2. 链接。在链接阶段将验证类中的字节码,为静态域分配存储空间,并且如果必须的话,将解析这个类创建的对其他类的所有引用。
3. 初始化、如果该类具有超类,则对其初始化,执行静态初始化器和静态初始化快。
Class initable = Initable.class;
System.out.println(initable.staitcFinal);
初始化有效地实现了尽可能的“惰性”。从对initable引用的创建中可以看到,仅使用.class语法来获得对类的引用不会引发初始化。但是为了产生Class引用,Class.forName()立即就进行了初始化,就像在initable3引用的创建中所看到的。
如果一个static final值是“编译器常量”,就像Initable.staticFinal那样,那么这个值不需要对Initable类进行初始化就可以被读取。但是,如果只是将一个域设置为static和final的,还不足以确保这种行为,例如,对Initable.staticFinal2的访问将强制进行类的初始化,因为它不是一个编译器常量。
如果一个static域不是final的,那么在对它访问时,总是要求在它被读取之前,要先进行链接和初始化,就像在对Initable2.staticNonFinal的访问中所看到的那样。
泛化的Class引用
Class引用总是指向某个Class对象,它可以制造类的实例,并包含可作用于这些实例的所有方法代码。它还包含该类的静态成员,因此,Class引用表示的就是它所指向的对象的确切类型,而该对象便是Class类的一个对象。
下面这两种语法都是正确的:
public static void main(String[] args) {Class intClass = int.class;Class<Integer> genericIntClass = int.class;genericIntClass = Integer.class;intClass = double.class;//genericIntClass = double.class //illegal}
public static void main(String[] args) {Class<FancyToy> ftClass = FancyToy.class;FancyToy fancyToy = ftClass.newInstance();Class<? super FancyToy> up = ftClass.getSuperclass();//this wont compile//Class<Toy> up2 = ftClass.getSuperclass();//Only produces Object;Object obj = up.newInstance();}
public static void main(String[] args) {Building b = new House();Class<House> houseType = House.class;House h = houseType.cast(b);h = (House)b;}