41.java编程思想——运行期类型鉴定语法

41.java编程思想——运行期类型鉴定语法

Java 用Class 对象实现自己的RTTI 功能——即便我们要做的只是象造型那样的一些工作。Class 类也提供了其他大量方式,以方便我们使用RTTI。

首先必须获得指向适当Class 对象的的一个句柄。一个办法是用一个字串以及Class.forName()方法。这是非常方便的,因为不需要那种类型的一个对象来获取Class 句柄。然而,对于自己感兴趣的类型,如果已有了它的一个对象,那么为了取得Class句柄,可调用属于Object 根类一部分的一个方法:getClass()。它的作用是返回一个特定的Class 句柄,用来表示对象的实际类型。

1     代码

interface HasBatteries {

}

interface Waterproof {

}

interface ShootsThings {

}

class Toy {

    // Comment outthe following default

    // constructorto see

    //NoSuchMethodError from (*1*)

    Toy() {

    }

    Toy(int i) {

    }

}

class FancyToy extends Toy implements HasBatteries,Waterproof, ShootsThings {

    FancyToy() {

        super(1);

    }

}

public class ToyTest {

    public staticvoidmain(String[] args){

        Class c = null;

        try {

            c = Class.forName("FancyToy");

        } catch (ClassNotFoundException e) {

        }

        printInfo(c);

        Class[] faces = c.getInterfaces();

        for (int i = 0; i < faces.length; i++)

            printInfo(faces[i]);

        Class cy = c.getSuperclass();

        Objecto = null;

        try {

            // Requires default constructor:

            o = cy.newInstance(); // (*1*)

        } catch (InstantiationException e) {

        } catch (IllegalAccessException e) {

        }

        printInfo(o.getClass());

    }

    static voidprintInfo(Class cc) {

        System.out.println("Class name: " + cc.getName() + " is interface? ["+ cc.isInterface()+ "]");

    }

} /// :~

2     执行

Classname: FancyToy is interface? [false]

Classname: HasBatteries is interface? [true]

Classname: Waterproof is interface? [true]

Classname: ShootsThings is interface? [true]

Classname: Toy is interface? [false]

从中可以看出,class FancyToy 相当复杂,因为它从Toy 中继承,并实现了HasBatteries,Waterproof 以及ShootsThings 的接口。在main()中创建了一个Class 句柄,并用位于相应try 块内的forName()初始化成FancyToy。

Class.getInterfaces 方法会返回Class 对象的一个数组,用于表示包含在Class 对象内的接口。若有一个Class 对象,也可以用getSuperclass()查询该对象的直接基础类是什么。当然,这种做会返回一个Class 句柄,可用它作进一步的查询。这意味着在运行期的时候,完全有机会调查到对象的完整层次结构。

若从表面看,Class 的newInstance()方法似乎是克隆(clone())一个对象的另一种手段。但两者是有区别的。利用newInstance(),我们可在没有现成对象供“克隆”的情况下新建一个对象。就象上面的程序演示的那样,当时没有Toy 对象,只有cy——即y 的Class 对象的一个句柄。利用它可以实现“虚拟构建器”。

换言之,我们表达:“尽管我不知道你的准确类型是什么,但请你无论如何都正确地创建自己。”在上述例子中,cy 只是一个Class 句柄,编译期间并不知道进一步的类型信息。一旦新建了一个实例后,可以得到Object 句柄。但那个句柄指向一个Toy 对象。当然,如果要将除Object 能够接收的其他任何消息发出去,首先必须进行一些调查研究,再进行造型。除此以外,用newInstance()创建的类必须有一个默认构建器。没有办法用newInstance()创建拥有非默认构建器的对象,所以在Java 1.0 中可能存在一些限制。然而,Java 1.1 的“反射”API(却允许我们动态地使用类里的任何构建器。

程序中的最后一个方法是printInfo(),它取得一个Class 句柄,通过getName()获得它的名字,并用interface()调查它是不是一个接口。

所以利用Class 对象,我们几乎能将一个对象的祖宗十八代都调查出来。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值