RTTI(Run-Time Type Identification,运行时类型识别)

RTTI(Run-Time Type Identification,运行时类型识别)

dynamic_cast

【格式】:dynamic_cast < type-id > ( expression)
该运算符把expression转换成type-id类型的对象。Type-id可以是类的指针、类的引用或者void*。如果type-id是类指针类型,那么expression也必须是一个指针,如果type-id是一个引用,那么expression也必须是一个引用。
【作用】:将一个基类对象指针(或引用)cast到继承类指针,dynamic_cast会根据基类指针 是否真正指向继承类对象来做相应处理, 即会作出一定的判断。
若对指针进行dynamic_cast,失败返回null,成功返回正常cast后的对象指针;
若对引用进行dynamic_cast,失败抛出一个异常,成功返回正常cast后的对象引用。
【注意】:
1、dynamic_cast在将父类cast到子类时, 父类必须要有虚函数,否则编译器会报错。
2、 dynamic_cast主要用于类层次间的上行转换和下行转换,还可以用于类之间的交叉转换。
在类层次间进行 上行转换时,dynamic_cast和static_cast的效果是一样的;
在进行下行转换时,dynamic_cast具有类型检查的功能,比static_cast更安全。
下面通过一个简单的例子来说明dynamic_cast的作用:
// 我是父类
class Tfather
{
public :
        virtual void f() { cout << "father's f()" << endl; }
};
// 我是子类
class Tson : public Tfather
{
public :
        void f() { cout << "son's f()" << endl; }
        int data; // 我是子类独有成员
};
int main()
{
        Tfather father;
        Tson son;
       son.data = 123;
        Tfather * pf;
        Tson * ps;
        /* 上行转换:没有问题,多态有效 */
       ps = &son;
       pf = dynamic_cast < Tfather *>(ps);
       pf->f();
        /* 下行转换(pf实际指向子类对象):没有问题 */
       pf = &son;
       ps = dynamic_cast < Tson *>(pf);
       ps->f();
       cout << ps->data << endl;            // 访问子类独有成员有效
        /* 下行转换(pf实际指向父类对象):含有不安全操作,dynamic_cast发挥作用返回NULL */
       pf = &father;
       ps = dynamic_cast < Tson *>(pf);
        assert (ps != NULL );                         // 违背断言,阻止以下不安全操作
       ps->f();
       cout << ps->data << endl;            // 不安全操作,对象实例根本没有data成员
        /* 下行转换(pf实际指向父类对象):含有不安全操作,static_cast无视 */
       pf = &father;
       ps = static_cast < Tson *>(pf);
        assert (ps != NULL );
       ps->f();
       cout << ps->data << endl;            // 不安全操作,对象实例根本没有data成员
       system( "pause" );
}

typeid

定义:typeid是C++的关键字之一,等同于sizeof这类的操作符。typeid操作符的返回结果是名为type_info的标准库类型的对象的引用。
如果表达式的类型是类类型且至少包含有一个虚函数,则typeid操作符返回表达式的动态类型(只在指针解引用和引用时体现),需要在运行时计算;否则,typeid操作符返回表达式的静态类型,在编译时就可以计算。
1.当typeid操作符的操作数是不带有虚函数的类类型时,typeid操作符会指出操作数的类型,而不是底层对象的类型。2.如果typeid操作符的操作数是至少包含一个虚拟函数的类类型时,并且该表达式是一个基类的引用或是指针解引用,则typeid操作符指出底层对象的派生类类型。

#include <iostream>
using namespace std;
class Base {
    virtual void f() {};
};
#define OUTPUT (f)     cout << #f << "\t: " << typeid (f).name() << endl;
class BaseA {};
class DeriveA : public BaseA {};
class BaseB
{
        virtual void f() {}
};
class DeriveB : public BaseB {};
int main()
{
       cout << "-------直接处理类名-------" << endl;
        OUTPUT ( BaseA );
        OUTPUT ( DeriveA );
        OUTPUT ( BaseB );
        OUTPUT ( DeriveB );
       cout << endl << "-------基类不含虚函数-------" << endl;
        BaseA baseA;
        DeriveA deriveA;
        OUTPUT (baseA);
        OUTPUT (deriveA);
        BaseA * pa;
       pa = &baseA;
        OUTPUT (*pa);
        OUTPUT (pa);
       pa = &deriveA;
        OUTPUT (*pa);
        OUTPUT (pa);
        BaseA & A1 = baseA;
        BaseA & A2 =deriveA;
        OUTPUT (A1);
        OUTPUT (A2);
       cout << endl << "-------基类含有虚函数-------" << endl;
        BaseB baseB;
        DeriveB deriveB;
        OUTPUT (baseB);
        OUTPUT (deriveB);
        BaseB * pb;
       pb = &baseB;
        OUTPUT (*pb);
        OUTPUT (pb);
       pb = &deriveB;
        OUTPUT (*pb);
        OUTPUT (pb);
        BaseB & B1 = baseB;
        BaseB & B2 = deriveB;
        OUTPUT (B1);
        OUTPUT (B2);
        return 0;
}

 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值