RTTI是所谓的运行时刻类型识别,它是OOP一个很重要的特性。我们知道,OOP编程中,派生类对象可以看作是一个基类对象,这种方式被成为UpCasting,即向上转型,这种转型是绝对安全的,其暗藏的机制在于,派生类对象中隐藏了一个基类对象,而这种绝对安全的转型方式,给我们带来了一个“多态”的好处:我们可以使用一个统一的接口来管理一个基类下面的不同派生类。在AO编程中,我们常常将一个函数的参数类型尽量放大,使得它的运行足够安全,例如Public function ConvertGeo(IGeometry pGeo)As IGeometry,哪怕这个方法只有处理Polygon的代码,却选择了传入一个IGeometry的参数。
当我们使用IGeometry的参数,这样避免了在将来改动这个函数的时候的许多麻烦,例如如果这个函数将来还可以处理Point、Polyline等类型对象,我们就不用修改函数的参数,因为无论是点、多义线还是多边形,都是IGeoemtry的一种,都可以看作是一种IGeometry。统一的接口解决了以后,我们再来看函数内部的情况。
处理Point和Polygon对象是不同的代码,那么,如何识别我们传入的对象是IPoint还是IPolygon呢?在VB中,我们使用的是TypeOf函数,如:
if TypeOf pGeo Is IPolygon then
......
这就是RTTI,即我们只有在程序的运行时刻才能识别我们传入的对象的具体类型,pGeo是以基类的身份传入的,但在具体处理的时候,我们需要识别它的具体身份才能具体处理。在java中也拥有类似的函数instanceof,意义是一样的,如对象obj是CClass的一个对象,即为:
if(obj instanceof CClass)
除此以外,java中还专门有个类Class,包含许多静态方法来帮助开发者获得一个对象在运行时刻的类型信息。
public static void main(String[] args) {
// TODO Auto-generated method stub
//以基类接口定义,
CFather cs1=new CSon1();
CFather cs2=new CSon2();
//将两个对象装入Object数组
Object[] arrs=new Object[2];
arrs[0]=cs1;
arrs[1]=cs2;
//下面的代码给arrs数组中的每个对象都产生一个类型信息记录对象,它是一个Class数组
Class[] types=new Class[arrs.length];
for(int i=0;i<arrs.length;i++){
types[i]=arrs[i].getClass();
}
for(int i=types.length-1;i>=0;i--){
/**if(arrs[i] instanceof CSon1){
System.out.println("cson1");
}else if(arrs[i] instanceof CSon2){
System.out.println("cson2");
}*/
for(int j=arrs.length-1;j>=0;j--){
//types[i]是Class对象,它有一个静态方法isInstance,用于判断对象arrs[j]是否为它的对象
if(types[i].isInstance(arrs[j])){
System.out.println(((Class)types[i]).getName());
}
}
}
}