1. C++有三个支持RTTI的元素。

  • 如果可能的话,dynamic_cast运算符将使用一个指向基类的指针来生成一个指向派生类的指针,否则,该运算符返回0——空指针。
  • typeid运算符返回一个对type_info对象的引用。
  • type_info结构存储了有关特定类型的信息。

只能将RTTI用于包含虚函数的类层次结构,原因在于只有对于这种类层次结构,才应该将派生对象的地址赋给基类指针。

2. 通常,如果指向的对象(*pt)的类型为Type或者是从Type直接或间接派生而来的类型,则下面的表达式将指针pt转换为Type类型的指针:

dynamic_cast<Type *> (pt)

否则,结果为0,即空指针。

3. typeid运算符使得能够确定两个对象是否为同种类型。它与sizeof有些相像,可以接受两种参数:

  • 类名
  • 结果为对象的表达式

typeid运算符返回一个对type_info对象的引用,其中,type_info是在头文件typeinfo中定义的一个类。type_info类重载了==和!=运算符,以便可以使用这些运算符来对类型进行比较。如果pg指向的是一个Magnificent对象,则下述表达式的结果为true,否则为false:

typeid(Magnificent) == typeid(*pg)

如果pg是一个空指针,程序将引发bad_typeid异常。该异常类型是从exception类派生而来的,是在头文件typeinfo中声明的。

type_info类的实现随厂商而异,但包含一个name()成员,该函数返回一个随实现而异的字符串,通常是类的名称。例如,下面的语句显示指针pg指向的对象所属的类定义的字符串:

cout << “Now processing type ” << typeid(*pg).name() << “.\n”;

#include <iostream>
#include <cstdlib>
#include <ctime>
#include <typeinfo> using namespace std; class Grand
{
private:
int hold;
public:
Grand(int h = 0) : hold(h){}
virtual void Speak() const { cout << "I am a grand class!\n"; }
virtual int Value() const { return hold; }
}; class Superb : public Grand
{
public:
Superb(int h = 0) :Grand(h){}
void Speak() const{ cout << "I am a Superb class!!\n"; }//const可以删除
virtual void Say() const
{
cout << "I hold the superb value of " << Value() << endl;
}
}; class Magnificent : public Superb
{
private:
char ch;
public:
Magnificent(int h = 0, char cv = 'A') : Superb(h), ch(cv){}
void Speak() const
{
cout << "I am a magnificent class!!!\n";
}
void Say() const
{
cout << "I hold the character " << ch << " and the integer " << Value() << "!" <<endl;
}
}; Grand * Getone(); int main()
{
srand(time(0));
Grand *pg;
Superb *ps;
for (int i = 0; i < 5; i++)
{
pg = Getone();
cout << "Now processing type " << typeid(*pg).name() << ".\n";
pg->Speak();
if (ps = dynamic_cast<Superb *>(pg))
ps->Say();
if (typeid(Magnificent) == typeid(*pg))
cout << "You are really Magnificent.\n";
}
return 0;
} Grand * Getone()
{
Grand *p;
switch (rand() % 3)
{
case 0:
p = new Grand(rand() % 100);
break;
case 1:
p = new Superb(rand() % 100);
break;
case 2:
p = new Magnificent(rand() % 100, 'A' + rand() % 26);
break;
default:
break;
} return p;
}

  运行得到结果如下:

最新文章

  1. web 开发入门
  2. 关于新书《修炼之道:.NET开发要点精讲》的各种说明
  3. mysql 主主互备
  4. 【转】 Nginx深入详解之多进程网络模型
  5. Python核心模块——urllib模块
  6. 2014 年10个最佳的PHP图像操作库--留着有用
  7. (转)iOS被开发者遗忘在角落的NSException-其实它很强大
  8. 浅析深究什么是SOA?(转)
  9. EasyInvoice 简介
  10. Old Sorting(转化成单调序列的最小次数,置换群思想)
  11. pig对null的处理(实际,对空文本处理为两种取值null或‘’)
  12. NI笔试——大数加法
  13. RobotFramework自动化测试环境配置
  14. springmvc注解驱动
  15. CSS优先级的及其衡量标准CSS权重
  16. Go 包管理工具--glide
  17. Linux上安装jdk,mysql
  18. PHP: Browser, Operating System (OS), Device, and Language Detect
  19. 通过Jekins执行bat脚本始终无法完成
  20. HDFS2.0架构以及HA详解

热门文章

  1. FIL怎么获得?FIL在哪里购买?
  2. 【小白学算法】5.链表(linked list)、链表的添加
  3. SynchronousQueue核心源码分析
  4. (原创)在Linux上安装运行Python3(CentOS7为例)
  5. 使用Amazon Pinpoint对用户行为追踪
  6. Distributed | MapReduce
  7. lustre文件系统环境搭建及测试
  8. Oracle recover current redo ORA-00600:[4193] (oracle 故障恢复current redo日志ORA-00600:[4193]报错)
  9. 浅入Kubernetes(7):应用部署实例,Deployment、Service、ReplicaSet
  10. JFX11+Maven+IDEA 发布跨平台应用的完美解决方案