运行时类型信息(Runtime Type Information
,RTTI
)使得你可以在程序运行时发现和使用类型信息。Java在运行时识别对象和类的信息,主要有两种方式:一种是传统的RTTI
,它假定我们在编译时已经知道了所有的类型;另一种是反射机制,它允许我们在运行时发现和使用类的信息。
RTTI的形式包含:
- 传统的类型转换,如“
(Shape)
”,由RTTI确保类型转换的正确性,如果执行了一个错误的类型转换,就会抛出一个ClassCastException
异常。 - 代表对象的类型的
Class
对象。通过查询Class
对象可以获取运行时所需的信息。 - 关键字
instanceof
。它返回一个布尔值,告诉我们对象是不是某个特定类型的实例。
Class对象
所有的类都是在对其第一次使用时,动态加载到JVM
中的,当程序创建第一个对类的静态成员的引用时,就会加载这个类。这个证明构造函数也是类的静态方法,即使在构造器之前并没有使用static
关键字。因此,使用new操作符创建类的新对象也会被当做对类的静态成员的引用。
因此,Java程序在它开始运行之前并非被完全加载,其各个部分是在必需时才加载的。
类加载器首先检查这个类的CLass
对象是否已经加载。如果尚未加载,默认的类加载器就会根据类名查找.class
文件。在这个类的字节码被加载时,它们会接受验证,以确保其没有被破坏,并且不包含不良Java代码。
一旦某个类的Class
对象被载入内存,它就被用来创建这个类的所有对象。
1 | class Candy { |
从输出中可以看到,Class
对象仅在需要的时候才被加载,static
初始化是在类加载时进行的。
类字面常量
Java还提供了另外一种方法来生成对Class
对象的引用,即使用类字面常量。
如: Gum.class
。
这样做不仅更简单,而且更安全,因为它在编译时就会受到检查。并且它根除了对forName()
方法的调用,所以也更高效。
当使用.class
来创建对Class对象的引用时,不会自动地初始化该Class
对象。为了使用类而做的准备工作实际包含三个步骤:
- 加载。这是由类加载器执行的。该步骤将查找字节码,并从这些字节码中创建一个
Class
对象。 - 链接。在链接阶段将验证类中的字节码,为静态域分配存储空间,并且如果必需的话,将解析这个类创建的对其他类的所有引用。
- 初始化。如果该类具有超类,则对其初始化,执行静态初始化器和静态代码块。
初始化被延迟到了对静态方法(构造器隐式地是静态的)或者非常数静态域进行首次引用时才执行。
泛化的Class引用
Class
引用总是指向某个Class
对象,它可以制造类的实例,并包含可作用于这些实例的所有方法代码。它还包含该类的静态成员,因此,Class
引用表示的就是它所指向的对象的确切类型,而该对象便是Class
类的一个对象。
Java SE5允许你对Class
引用所指向的Class
对象的类型进行限定,用到了泛型语法。
1 | public class GenericClassReferences { |
instanceof与Class的等价性
在查询类信息时,以instanceof
的形式(即以instanceof
的形式或isInstance()
的形式,它们产生相同的结果)与直接比较Class
对象有一个很重要的差别。
1 | package typeinfo |
instanceof
和isInstance()
生成的结果完全一样,equals()
和==
生成的结果完成一样。instanceof
判断某个对象是否是指定类或者指定类的子类。而equals()
比较的是实际的Class
对象,不考虑继承。