文章目录
第14章 类型信息
运行时类型信息使你可以在程序运行时发现和使用类型信息,使你从只能在编译期执行面向操作的禁锢中解脱出来。一种是传统的RTTI,一种是反射机制。
14.1为什么需要RTTI
前面讲到的多态,当把Shape对象放入List<Shape>的数组时会向上转型。但在向上转型的时候也丢失了Shape对象的具体类型。
当从数组中取出元素时,会自动将结果转型回Shape。这是RTTI最基本的使用形式。
接下来就是多态的机制了,Shape对象实际执行什么样的代码,是由引用所指向的具体对象Circle、Square或者Triangle决定的。
14.2 Class对象
类型信息在运行时是如何表示的
Class对象就是用来创建类的所有的对象的。Java使用Class对象来执行其RTTI,即使你正在执行的是类似转型这样的操作。
每个类都有一个Class对象,为了生成这个对象,运行这个程序的JVM将使用被称为“类加载器”的子系统。原生类加载器加载的是所谓的可信类,包括Java API类,它们通常是从本地盘加载的。
所有的类都是在对其第一次使用时,动态加载到JVM中的。当程序创建第一个对类的静态成员的引用时,就会加载这个类。
Class.forName();
forName()是取得Class对象的引用的一种方法。它是用一个包含目标类的文本名的String作为参数,返回的是一个Class对象的引用。
无论何时,只要你想在运行时使用类型信息,就必须首先获得Class对象的引用。
14.2.1类字面常量
Java还提供了另一种方法来生成对Class对象的引用,即使用类字面常量。不会引发初始化。
类名.class;
14.2.2泛化的Class引用
普通的类引用可以被重新赋值为指向任何其他的Class对象,但泛型不行。
除非用通配符?
Class<?> intClass = int.class;
intClass = double.class;
或者
Class<? extends Number> bounded = int.class;
bounded = double.class;
bounded = Number.class;
因为Integer Class对象不是Number Class对象的子类,必须加?extends
14.3类型转换前先做检查
instanceof
14.6反射:运行时的类信息
基于构件的编程
远程方法调用(RMI)
Class类与java.lang.reflect类库一起对反射的概念进行了支持,该类库包含了Field、Method以及Constructor类(每个类都实现了Member接口).
对于反射机制来说,.class文件在编译时是不可获取的,所以是在运行时打开和检查.class文件。
14.6.1类方法提取器
反射在Java中是用来支持其他特性的,例如对象序列化和JavaBean。但是如果能动态地提取某个类的信息还是很有用的。请考虑类方法提取器。
14.7动态代理
代理是基本的设计模式之一,创建用来代替实际对象的对象,这些操作通常涉及与实际对象的通信。
interface Interface{
void doSomething();
void somethingElse(String arg);
}
class RealObject implements Interface{
public void doSomething(){
print("doSomething");
}
public void somethingElse(String arg){
print("somethingElse "+arg);
}
}
class SimpleProxy implements Interface{
private Interface proxied;
public SimpleProxy(Interface proxied){
this.proxied = proxied;
}
public void doSomething(){
print("SimpleProxy doSomething");
proxied.doSomething();
}
public void somethingElse(String arg){
print("SimpleProxy somethingElse "+arg);
proxied.somethingElse(arg);
}
}
class SimpleProxyDemo{
public static void consumer(Interface iface){
iface.doSomething();
iface.somethingElse("bonobo");
}
public static void main(String[] args){
consumer(new RealObject());
consumer(new SimpleProxy(new RealObjec());
}
}
consumer()接受Interface,无法知道正在获得的到底是RealObject还是SimpleProxy。但是SimpleProxy已经被插入到了客户端和RealObject之间。
Java的动态代理可以动态地创建代理并动态地处理对所代理方法的调用。