读书笔记:《JAVA编程思想(Thinking in Java)》第14章 类型信息

第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的动态代理可以动态地创建代理并动态地处理对所代理方法的调用。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值