/*********************************************************************************
*   发布日期:2017-11-13 14:01:16
*   进度:
*   作者:LuoTian
*   备注:运行时类型识别
*********************************************************************************/

这是读《深入浅出》做的笔记,书本程序是对MFC框架的仿真,我做的是对他的程序仿真,主要为了更进一步的简化理解。

为了简化问题,我这里只用三个类A,B,C说明问题,继承关系为A<-B<-C,即A为根基类,A派生B,B派生C,层次非常简单,现在做一下通俗的解释。

思想:

问题的引出是判断某类是否为某个类的派生类,比如C类明显是B的派生类,也即B为C的基类,如何让这种派生关系记录下来呢?于是采用一种直接了当的办法,在每个类中添加一个信息结构体,专门记录各种信息。
比如,B中添加一个信息结构体,包括如下信息:

  • 我的类名是什么?->B
  • 我的基类是谁?->类A(实际为类A的信息结构体地址)
  • 我的前驱类是谁(pNext)?->类A(类A的信息结构体地址)
  • 我的类大小?->sizeof(B)

等等,可以看出,此结构体不仅记录自身的信息,还记录和外界的相互关系。
其中,A类是根类,所以它的基类和前驱都是NULL。

在这个思想引导下,通过一系列操作之后,形成的形状就和链表非常相似(参看文章末尾),在此"链表"的末端有一个CRuntimeClass类型的指针pFirst,通过这个pFirst就可以到达各个类的CRuntimeClass。

从思路中可以看出,关键部分就是安插在每个类中的结构体CRuntimeClass,而且思路也很简单。

《深入浅出》模仿了MFC的宏实现,我这个代码是把宏解开之后进行的模仿,主要便于观察,只要用宏再包装一下就可以逆转回去。

示例代码,功能:在ABC三个类中各安装一个CRuntimeClass并链接起来,从尾端开始遍历各个类的名字。

#include <iostream>
using namespace std; //CRuntimeClass结构,并初始化静态成员pFirst=NULL;
struct CRuntimeClass
{
char * name; //类名
int ClassSize; //类大小;
CRuntimeClass *pBase;//基类的CRuntimeClass地址 static CRuntimeClass *pFirst;
CRuntimeClass *pNext;
};
CRuntimeClass * CRuntimeClass::pFirst=NULL;//①
//****************结束****************************** struct AFX_CLASSINIT
{
AFX_CLASSINIT(CRuntimeClass *pNewClass)
{
pNewClass->pNext=CRuntimeClass::pFirst;
CRuntimeClass::pFirst=pNewClass;
}
}; class A //根基类;
{
public:
static CRuntimeClass classA;
virtual CRuntimeClass * GetRuntimeClass()const
{
return &A::classA;
}
}; class B:public A //B类
{
public:
static CRuntimeClass classB;
virtual CRuntimeClass * GetRuntimeClass()const
{
return &B::classB;
}
};
class C:public B //C类
{
public:
static CRuntimeClass classC;
virtual CRuntimeClass * GetRuntimeClass()const
{
return &C::classC;
}
}; //***********************实现部分*************************// struct CRuntimeClass A::classA={ //A类的CRuntimeClass初始化
"classA类",
sizeof(A),
NULL,//根基类A由于没有基类,所以Base为NULL;
};
static AFX_CLASSINIT _init_A(&A::classA);//② struct CRuntimeClass B::classB={ //B类的CRuntimeClass初始化
"classB类",
sizeof(B),
&A::classA,
};
static AFX_CLASSINIT _init_B(&B::classB);//③ struct CRuntimeClass C::classC={ //C类的CRuntimeClass初始化
"classC类",
sizeof(C),
&B::classB,
};
static AFX_CLASSINIT _init_C(&C::classC);//④ int main(int argc,char *argv[])
{
CRuntimeClass *p;
for(p=CRuntimeClass::pFirst;p!=NULL;p=p->pNext)
{
cout<<p->name<<endl;
}
delete p;
return ;
}

代码运行到①时,pFirst=NULL;
代码运行到②时,pFirst指向类A的CRuntimeClass;
代码运行到③时,pFirst指向类B的CRuntimeClass;
代码运行到④时,pFirst指向类C的CRuntimeClass;

《深入浅出MFC》系列之运行时类型识别(RTTI)的更多相关文章

  1. c++运行时类型识别(rtti)

    一个简单运行时类型识别 namespace rtti_ex { /* * 类型信息基类 */ class i_type_info { public: // 判断是否是指定类型 bool is(cons ...

  2. C++之运行时类型识别RTTI

     C++ Code  12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849 ...

  3. C++学习之显式类型转换与运行时类型识别RTTI

    static_cast const_cast reinterpret_cast 运行时类型识别(RTTI) dynamic_cast 哪种情况下dynamic_cast和static_cast使用的情 ...

  4. 运行时类型识别RTTI

    1.RTTI的工作原理 例1. 用Class加载对象示例. package RTTI; public class Candy { static{ System.out.println("Lo ...

  5. C++——运行时类型识别RTTI

    1.实现方式 typeid运算符,返回表达式的类型 dynamic_cast运算符,基类的指针或引用安全地转换成派生类的指针或引用 2.适用于:使用基类的指针或引用执行派生类的操作,且该操作不是虚函数 ...

  6. Java基础之RTTI 运行时类型识别

    运行时类型识别(RTTI, Run-Time Type Identification)是Java中非常有用的机制,在Java运行时,RTTI维护类的相关信息. 多态(polymorphism)是基于R ...

  7. MFC六大核心机制之二:运行时类型识别(RTTI)

    上一节讲的是MFC六大核心机制之一:MFC程序的初始化,本节继续讲解MFC六大核心机制之二:运行时类型识别(RTTI). typeid运算子 运行时类型识别(RTTI)即是程序执行过程中知道某个对象属 ...

  8. 框架原理第二讲,RTTI,运行时类型识别.(以MFC框架讲解)

    框架原理第二讲,RTTI,运行时类型识别.(以MFC框架讲解) 一丶什么是RTTI,以及RTTI怎么设计 通过第一讲,我们知道了怎么样升成一个窗口了,以及简单的消息循环. 第二讲则是主要讲解RTTI ...

  9. MFC原理第三讲.RTTI运行时类型识别

    MFC原理第三讲.RTTI运行时类型识别 一丶什么是RTTI RTTI. 运行时的时候类型的识别. 运行时类型信息程序.能够使用基类(父类)指针 或者引用 来检查这些指针或者引用所指的对象. 实际派生 ...

随机推荐

  1. 微信小程序 wxml中的属性记录

    1. view 标签中的属性 style 中的参数 margin-top:10px;  (向上距离) display : flex;  (display : flex 容器声明) flex-direc ...

  2. 瑞波Ripple-OnlineDeletion配置的理解

    1.概要 如果没有设置online deletion,那么Rippled Server的本地数据库大小将会无限增长.为了释放磁盘空间,那种情况应该被制止并且那些数据库文件应该被手动删除.online ...

  3. devexpress之barManager 使用

    这次我不想使用ribboncontrol 控件 作为窗口菜单栏,也不想用传统的那种字体的方式 标题栏 一.Bars 1.   把BarManager组件添加到窗体中后,会自动创建两个空的 bars: ...

  4. L1-030 一帮一(15)(代码)

    L1-030 一帮一(15 分) "一帮一学习小组"是中小学中常见的学习组织方式,老师把学习成绩靠前的学生跟学习成绩靠后的学生排在一组.本题就请你编写程序帮助老师自动完成这个分配工 ...

  5. MySQL学习笔记-锁相关话题

    在事务相关话题中,已经提到事务隔离性依靠锁机制实现的.在本篇中围绕着InnoDB与MyISAM锁机制的不同展开,进而描述锁的实现方式,多种锁的概念,以及死锁产生的原因.   Mysql常用存储引擎的锁 ...

  6. UI设计教程学习分享:APP布局

    一.宫格布局 这种APP信息布局方式也是我们目前最常见的一种方式,也是符合用户习惯和黄金比例的设计方式,最知名的就是锤子手机的界面设计.锤子手机界面设计欣赏知名的APP设计采用的九宫格.六宫格等方式布 ...

  7. 存储过程和函数 PROCEDURE & FUNCTION

    SQL语句执行的时候,要首先编译,然后在被执行.在大型数据库系统中,为了提高效率,将为了完成特定功能的SQL语句集进行编译优化后,存储在数据库服务器中,用户通过指定存储过程的名字来调用执行. 具体而言 ...

  8. ros pluginlib 段错误

    最近在重新回看ROS插件时,运行出现了段错误,发现是boost版本问题,我目前版本是1.66,应该调整至1.58版本,如果跟其他软件使用不同的boost版本时,可以把相应版本编译到本地,不instal ...

  9. Android中fragment之间和Activity的传值、切换

    功能介绍:通过一个activity下方的三个按钮,分别是发送消息(sendButton).聊天记录(chatButton).常用语(commonButton).当单击按钮是,来切换上方的fragmen ...

  10. Spark学习笔记-GraphX-1

    Spark学习笔记-GraphX-1 标签: SparkGraphGraphX图计算 2014-09-29 13:04 2339人阅读 评论(0) 收藏 举报  分类: Spark(8)  版权声明: ...