转自:http://blog.chinaunix.net/uid-28541347-id-4251713.html

从RTTI谈C++的向下转型

1.什么是RTTI?

RTTI 是“Runtime Type Information”的缩写,意思是:运行时类型信息。它提供了运行时确定对象类型的方法。允许“用指向基类的指针或引用来操纵对象”的程序能够获取到“这些指针或引用所指对象”的实际派生类型。在 c++中,为了支持 RTTI 提供了两个操作符 : 
    1 dynamic_cast 操作符:它允许在运行时刻进行类型转换,从而使程序能够在一个类层次结构中安全地转换类型(安全的向下转型)。把基类指针转换成派生类指针,或把指向基类的左值转换成派生类的引用。当然只有在保证转换能够成功的情况下才可以;  
    2 typeid操作符:它指出指针或引用指向的对象的实际派生类型。  
    但是,对于要获得的派生类类型的信息,dynamic_cast 和 typeid操作符的操作数的类型必须是带有一个或多个虚拟函数的类类型,即对于带有虚拟函数的类而言,RTTI 操作符是运行时刻的事件,而对于其他类而言,它只是编译时刻的事件。

2. RTTI如何实现

C++的RTTI是最简单的,只能获得类名和相关的继承信息;而VB、Delphi、Java等确复杂得多,甚至于支持属性名、方法名、事件名等。

如图,C++使用type_info类的对象保存每个对象的相关信息如对象名称、对象类型等。而所谓RTTI就是在执行期取得对象的type_info。所以C++采用了和虚函数同样的办法,使用vtable(通常为第一个slot)保存需要执行期获得type_info的对象的type_info对象地址。那么由pt指向的class object的type_info可在执行期通过如下方式获得:

(type_info)*(type_info*)(pt->vptr[0]);

通过上述实现分析我们也可以知道,C++的RTTI只能用于那些展现“多态”(内含虚函数)的类型有效。因为只有这样的类才有vtable。

3. 向下转型

将一个Base class对象的指针转为Derived class对象的指针或将Base class的左值转为Derived class的引用称为向下转型。

注:不能讲Base class的对象转为Derived class的对象(除非定义相应转换操作符)。如下语句是错误的:

Base bobj;

Derived dobj=static_cast(bobj) ;//error,

Derived dobj=(Derived)(bobj) ;//error

但如下语句是正确的:

Derived dobj;

Base bobj=(Base)dobj;//正确,造成对象切割

Base bobj=static_cast(dobj);//正确,造成对象切割

向下转型示例:

(1) 基类指针转为子类指针

Base bobj;

Base *pb=&bobj;

Derived *pd=(Derived*)pb;//方式一

Derived *pd=static_cast(pb);//方式二

Derived *pd=dynamic_cast(pb);//方式三,只有Base class中有虚函数才能使用,否则编译错误(Derived class中有虚函数也不行)

(2) 基类左值转子类引用

Base bobj;

Derived dobj;

Derived &dref=(Derived&)(bobj);//方式一

Derived &dref=static_cast(bobj);//方式二

Derived &dref=dynamic_cast(bobj);//方式三,只有Base class中有虚函数才能使用,否则编译错误(Derived class中有虚函数也不行)

(3) 基类左值转子类引用

Derived dobj;

Base& bref=dobj;

Derived &dref=(Derived&)(bref);

Derived &dref=static_cast(bref);

Derived &dref=dynamic_cast(bref);

向下转型的隐患

向下转型有着潜在的危险,因为当基类指针指向的是基类对象,而将基类指针转为子类指针时,如果通过转换后的子类指针访问子类的专有成员,就会造成内存错误,因为实际指向的是基类对象,而基类对象中不存在这些成员。(引用转换类似)

4. 安全的向下转型——dynamic_cast<>()

所谓“安全的向下转型”即只有当Base class的指针确实指向Derived class对象时才能将其转为Derived class的指针。但是我们知道Base class的指针指向的对象类型在执行期是可以改变的(指向Base class对象或Derived class对象),所以要想保证向下转型的安全性,就必须在执行期对Base class的指针有所查询,看看它所指向对象的真正类型。

dynamic_cast运算符可以在执行期确定指针指向的真正类型(前提是Base class中要有虚函数)。当对Base class的指针向下转型时,如果向下转型时是安全的(也就是Base type pointer指向一个Derived class object),则这个运算符传回相应的Derived class指针,如果向下转型是不安全的,则这个运算符传回0.

当对Base class的引用向下转型时,如果向下转型时是安全的(也就是Base type reference引用一个Derived class object),则这个运算符传回相应的Derived class引用,否则抛出一个bad_cast exception,而不是返回0. 不返回0的原因是,若将一个引用设为0,会使一个临时性对象产生出来,该临时性对象的初值为0,这个引用被设置为这个对象的别名。

dynamic_cast的成本

由于要在执行期获取对象类型信息(type_info),类似虚函数调用:

(type_info)*(type_info*)(pt->vptr[0]);

当使用dynamic_cast进行Base class指针向下转型时,dynamic_cast会采用这种方法获取Base class指针和Derived class指针指向对象的type_info对象,并将两个type_info对象中的类型描述器交给一个runtime library函数,比较之后告诉我们是否吻合。这显然比static_cast的开销昂贵的多,但是安全的多。

5. Typeid运算符

Typeid运算符传回一个type_info对象的const引用。

1. typeid操作符必须与表达式或类型名一起使用。例如,内置类型的表达式和常量可以被用作 typeid的操作数。

(1) 当操作数不是类类型时,typeid操作符会指出操作数的类型 ,此时type_info对象的引用是在编译期获得,如: 
        int iobj; 
        cout << typeid( iobj ).name() << endl; // 打印: int 
        cout << typeid( 8.16 ).name() << endl; // 打印: double

(2) 当typeid操作符的操作数是类类型,但不是带有虚拟函数的类类型时,typeid操作符会指出操作数的类型,而不是底层对象的类型,此时type_info对象的引用是在编译期获得,如:

class Base { /* 没有虚拟函数 */ }; 
       class Derived : public Base { /* 没有虚拟函数 */ }; 
        Derived dobj; 
       Base *pb = &dobj; 
       cout << typeid( *pb ).name() << endl; // 打印: class Base

由于typeid操作符的操作数是 Base 类型的,即表达式*pb 的类型。而 Base 不是一个带有虚拟函数的类类型,所以typeid的结果是Base。尽管 pb 指向的底层对象的类型是 Derived。

(3) 当typeid操作符的操作数是类类型,且该类是带有虚拟函数的类类型时,typeid操作符会指出实际底层对象的类型,此时type_info对象的引用是在执行期获得,如:

class Base { /* 有虚拟函数 */ };

class Derived : public Base {};

Derived dobj;

Base *pb = &dobj;

cout << typeid( *pb ).name() << endl; // 打印: class Derived

2. 可以对 typeid的结果进行比较。例如  
     #include <type_info> 
     employee *pe = new manager; 
     employee& re = *pe; 
 
     if ( typeid( pe ) == typeid( employee* ) ) // true 
     if ( typeid( pe ) == typeid( manager* ) ) // false 
     if ( typeid( pe ) == typeid( employee ) ) // false 
     if ( typeid( pe ) == typeid( manager ) ) // false 
     if语句的条件子句比较“在一个表达式上应用typeid操作符的结果”和“用在类型名操作数上的typeid操作符的结果”。注意比较  
     typeid( pe ) == typeid( employee* )  的结果为true。

这是因为操作数pe 是一个指针,而不是一个类类型。为了要获取到派生类类型 ,typeid的操作数必须是一个类类型(带有虚拟函数)。表达式 typeid(pe)指出pe 的类型,即指向employee 的指针,它与表达式 typeid(employee*)相等。而其他比较的结果都是false。  
    当表达式*pe 被用在typeid上时,结果指出pe 指向的底层对象的类型  
    typeid( *pe ) == typeid( manager ) // true 
    typeid( *pe ) == typeid( employee ) // false 
      在这两个比较中,因为*pe 是一个类类型的表达式,该类带有虚拟函数,所以typeid的结果指出操作数所指的底层对象的类型,即manager 。 
   typeid操作符也可以被用在引用上。例如  
   typeid( re ) == typeid( manager ) // true 
   typeid( re ) == typeid( employee ) // false 
   typeid( &re ) == typeid( employee* ) // true 
   typeid( &re ) == typeid( manager* ) // false

在前两个比较中,操作数re 是带有虚拟函数的类类型。因此 typeid操作数的结果指出 re指向的底层对象的类型。在后两个比较中,操作数&re 是一个类型指针,因此 typeid操作符的结果指出操作数的类型,即 employee* 。

从RTTI谈C++的向下转型的更多相关文章

  1. Java入门记(二):向上转型与向下转型

    在对Java学习的过程中,对于转型这种操作比较迷茫,特总结出了此文.例子参考了<Java编程思想>. 目录 几个同义词 向上转型与向下转型 例一:向上转型,调用指定的父类方法 例二:向上转 ...

  2. java 向上,向下转型

    在对Java学习的过程中,对于转型这种操作比较迷茫,特总结出了此文.例子参考了<Java编程思想>. 目录 几个同义词 向上转型与向下转型 例一:向上转型,调用指定的父类方法 例二:向上转 ...

  3. Java多态之向下转型

    目录 Java多态之向下转型 强制类型转换 instanceof Java多态之向下转型 往期回顾:我们学习了向上转型和动态绑定的概念,可以知道在继承关系中,将一个子类对象赋值给父类的引用变量,调用父 ...

  4. java 向上转型 向下转型

    //父类 四边形 class Quadrangle{ public static void draw (Quadrangle q){ } } //子类  public class Parallelog ...

  5. Java向上转型与向下转型

    一.向上转型 例如:Parent p=new Son(); 这样引用p只能调用子类中重载父类的方法:但属性是父类的:如果想调用子类属性的话,可以用getter()方法. 二.向下转型 子类对象的父类引 ...

  6. (转载)java多态(2)-------Java转型(向上或向下转型)

    5.13.1 向上转型 我们在现实中常常这样说:这个人会唱歌.在这里,我们并不关心这个人是黑人还是白人,是成人还是小孩,也就是说我们更倾向于使用抽象概念“人”.再例如,麻雀是鸟类的一种(鸟类的子类), ...

  7. Java中的向上转型和向下转型

    首先要明白一点向上转型和向下转型他们都是建立在继承的基础上. 一.向上转型 子类到父类的转换通常称作向上转型,通俗的说就是定义父类对象指向子类对象. 下面通过一个例子来深入理解向上转型. //定义一个 ...

  8. 集合 ArrayList 向下转型 遍历

    List  list=new ArrayList(); Person p1=new Person("lisi1",21); Person p2=new Person("l ...

  9. C++ Primer 学习笔记_32_STL实践与分析(6) --再谈string类型(下)

    STL实践与分析 --再谈string类型(下) 四.string类型的查找操作 string类型提供了6种查找函数,每种函数以不同形式的find命名.这些操作所有返回string::size_typ ...

随机推荐

  1. linux安装oracle的官方文档

    1:https://docs.oracle.com/cd/E11882_01/install.112/e47689/toc.htm 2:https://oracle-base.com/articles ...

  2. 【Linux_Unix系统编程】Chapter10 时间

    chapter10 时间 1:真实时间:度量这一时间的起点有二:(1)某个标准点:(2)进程生命周期内的某个固定时点(通常为程序启动) 2:进程时间:一个进程所使用的CPU时间总量,适用于对程序,算法 ...

  3. 本地管理表空间和字典管理表空间的特点,ASSM有什么特点

    字典管理表空间(Dictionary-Managed Tablespace简称DMT),8i以前包括以后都还可以使用的一种表空间管理模式,通过数据字典管理表空间的空间使用. Oracle使用两个字典来 ...

  4. 初学 python 之 用户登录实现过程

    要求编写登录接口 : 1. 输入用户名和密码 2.认证成功后显示欢迎信息 3.用户名输错,提示用户不存在,重新输入(5次错误,提示尝试次数过多,退出程序) 4.用户名正确,密码错误,提示密码错误,重新 ...

  5. 分割List为指定size

    背景 老项目,用的原生的JDBC,获取连接,预编译...然后业务需要要更新很多条数据,我就写了条件为 ——IN()... 根据传入的 list 的 size 循环的给sql语句拼接上“ ? ”为了之后 ...

  6. python编程之禅

    在python界面输入 import this >>> import this The Zen of Python, by Tim Peters Beautiful is bette ...

  7. 加密算法之AES算法(转)

    转载http://www.mamicode.com/info-detail-514466.html 0 AES简介 美国国家标准技术研究所在2001年发布了高级加密标准(AES).AES是一个对称分组 ...

  8. 前端-javascript-BOM-浏览器对象模型

    BOM的介绍---浏览器对象模型. 操作浏览器部分功能的API.比如让浏览器自动滚动. -------------------------------------------------------- ...

  9. ABAP-动态编程

    转载:http://www.cnblogs.com/jiangzhengjun/p/4293407.html 动态编程 动态的基本语法 多种不同的动态编程 动态字段 动态类型 指定结构.内表组件字段的 ...

  10. UI5-文档-4.9-Component Configuration

    在我们介绍了模型-视图-控制器(MVC)概念的所有三个部分之后,现在我们将讨论SAPUI5的另一个重要的结构方面. 在这一步中,我们将把所有UI资产封装在一个独立于索引的组件中.html文件.组件是S ...