MFC CObject浅析

1.CObject简要声明

2.CRuntimeClass结构

3.RUNTIME_CLASS

4.DYNAMIC支持

5.DYNCREATE支持

6.SERIAL支持

CObject是大部分的MFC类的基类

为了完成MFC类的判断、动态生成、序列化等特殊功能,CObject中添加了特定的处理。

为了进一步增强对MFC类对象的理解,在此对CObject源码及相关宏定义进行分析。

(所附代码并非原始代码,为说明问题而作了删减。)

1.CObject简要声明

class CObject
{
public:
 virtual CRuntimeClass* GetRuntimeClass() const;
 virtual ~CObject(); 
 
 void* PASCAL operator new(size_t nSize);
 void* PASCAL operator new(size_t, void* p);
 void PASCAL operator delete(void* p);
 void PASCAL operator delete(void* p, void* pPlace);

#if defined(_DEBUG) //调试模式用,多了nLine参数,用于保存原码行号。
 void* PASCAL operator new(size_t nSize, LPCSTR lpszFileName, int nLine);
 void PASCAL operator delete(void *p, LPCSTR lpszFileName, int nLine);
#endif
protected:
 CObject();
private:
 CObject(const CObject& objectSrc);
 void operator=(const CObject& objectSrc);
// Attributes
public:
 BOOL IsSerializable() const;
 BOOL IsKindOf(const CRuntimeClass* pClass) const;
// Overridables
 virtual void Serialize(CArchive& ar);
#if defined(_DEBUG) //调试模式下用
 virtual void AssertValid() const;
 virtual void Dump(CDumpContext& dc) const;
#endif
public:
 static const AFX_DATA CRuntimeClass classCObject;
 static CRuntimeClass* PASCAL _GetBaseClass();
};
在此声明中很多都是纯虚函数,定义的一个一般对象的"界面"

2.CRuntimeClass结构

在CObject中包含一个静态成员变量,

static CRuntimeClass classCObject;它是MFC内部用来管理类的重要结构,记录了很多对象所属类的重要信息,通过它在运行时完成对类的管理。

很多内部管理成员函数及宏定义都建立在CRuntimeClass的基础上的。

struct CRuntimeClass
{
 //类名称
 LPCSTR m_lpszClassName;
 //大小
 int m_nObjectSize;
 //版本
 UINT m_wSchema; 
 CObject* (PASCAL* m_pfnCreateObject)(); // NULL => abstract class
 //指向基类CRuntimeClass的指针,用于在运行时记录类继承关系。
#ifdef _AFXDLL
 CRuntimeClass* (PASCAL* m_pfnGetBaseClass)();
#else
 CRuntimeClass* m_pBaseClass;
#endif
// Operations
 //建立对象
 CObject* CreateObject();
 //派生判断
 BOOL IsDerivedFrom(const CRuntimeClass* pBaseClass) const;
// Implementation
 //存储
 void Store(CArchive& ar) const;
 //读入
 static CRuntimeClass* PASCAL Load(CArchive& ar, UINT* pwSchemaNum);
 // CRuntimeClass objects linked together in simple list
 CRuntimeClass* m_pNextClass;       // linked list of registered classes
};
3.RUNTIME_CLASS

RUNTIME_CLASS(class_name)用于返回指向运行时类信息结构的指针

定义如下:

#define RUNTIME_CLASS(class_name) ((CRuntimeClass*)(&class_name::class##class_name))
4.DYNAMIC支持

在CObject 派生类中,可以获得动态"验证"支持,访问运行时类信息

方法:

声明时添加宏:DECLARE_DYNAMIC( class_name )

实现时添加宏: IMPLEMENT_DYNAMIC

原码分析:

DECLARE_DYNAMIC(class_name)相当于在类中添加如下声明

protected: 
 static CRuntimeClass* PASCAL _GetBaseClass(); 
public: 
 //静态成员CRuntimeClass,给此派生类添加了运行时类信息,这样就可以使用CRuntimeClass成员判断类信息了。
 //此成员名字格式为"class"+"类名",RUNTIME_CLASS()宏就是返回此结构的指针
 static const AFX_DATA CRuntimeClass class##class_name; 
 virtual CRuntimeClass* GetRuntimeClass() const; 
IMPLEMENT_DYNAMIC:

#define IMPLEMENT_DYNAMIC(class_name, base_class_name)  IMPLEMENT_RUNTIMECLASS(class_name, base_class_name, 0xFFFF, NULL)
#define IMPLEMENT_RUNTIMECLASS(class_name, base_class_name, wSchema, pfnNew)  //返回基类运行时信息结构的指针
 CRuntimeClass* PASCAL class_name::_GetBaseClass()   { return RUNTIME_CLASS(base_class_name); }  //初始化本类的运行时信息,依次为类名、大小,版本 ,NULL,基类
 AFX_COMDAT const AFX_DATADEF CRuntimeClass class_name::class##class_name = {   #class_name, sizeof(class class_name), wSchema, pfnNew,    &class_name::_GetBaseClass, NULL };  //返回运行时类信息,重载了CObject的GetRuntimeClass,使得CObject中声明的接口对具体的派生类有效
 CRuntimeClass* class_name::GetRuntimeClass() const   { return RUNTIME_CLASS(class_name); } 有了这些,就可以使用RUNTIME_CLASS()宏,以及用BOOL IsKindOf( const CRuntimeClass* pClass ) const判断类类型了。

5.DYNCREATE支持

类的实例动态生成支持

方法:

添加声明:DECLARE_DYNCREATE( class_name )

添加实现:IMPLEMENT_DYNCREATE( class_name, base_class_name )

原码分析:

DECLARE_DYNCREATE( class_name )

#define DECLARE_DYNCREATE(class_name)  //具有DYNAMIC支持
 DECLARE_DYNAMIC(class_name)  //对象建立支持
 static CObject* PASCAL CreateObject();
IMPLEMENT_DYNCREATE(class_name, base_class_name):

#define IMPLEMENT_DYNCREATE(class_name, base_class_name)  //动态建立对象
 CObject* PASCAL class_name::CreateObject()   { return new class_name; }  //填写运行时类信息,与DYNAMIC不同的是,有pfnNew参数 
 IMPLEMENT_RUNTIMECLASS(class_name, base_class_name, 0xFFFF,   class_name::CreateObject)
6.SERIAL支持

将对象储存,以及读取建立对象支持

方法:

添加声明:DECLARE_SERIAL( class_name )

添加实现:IMPLEMENT_SERIAL( class_name, base_class_name, wSchema )

原码分析

DECLARE_SERIAL( class_name ):

#define DECLARE_SERIAL(class_name)  //动态生成支持
 _DECLARE_DYNCREATE(class_name)  //文档操作符
 AFX_API friend CArchive& AFXAPI operator>>(CArchive& ar, class_name* &pOb);
IMPLEMENT_SERIAL(class_name, base_class_name, wSchema):

#define IMPLEMENT_SERIAL(class_name, base_class_name, wSchema)  //动态生成支持
 CObject* PASCAL class_name::CreateObject()   { return new class_name; }  //填写运行时类信息,包括版本号,生成函数指针 
 _IMPLEMENT_RUNTIMECLASS(class_name, base_class_name, wSchema,   class_name::CreateObject)  AFX_CLASSINIT _init_##class_name(RUNTIME_CLASS(class_name));  //文档支持实现
 CArchive& AFXAPI operator>>(CArchive& ar, class_name* &pOb)   { pOb = (class_name*) ar.ReadObject(RUNTIME_CLASS(class_name));    return ar; } 在派生类中重载virtual void Serialize(CArchive& ar);以实现类数据的保存及建立后读入。

从而实现类的保存,及读入动态建立。

本文来自CSDN博客,转载请标明出处:http://blog.csdn.net/FMD/archive/2001/06/16/5526.aspx

http://blog.csdn.net/akof1314/article/details/5597644

MFC浅析(4) CObject浅析的更多相关文章

  1. 阻塞和唤醒线程——LockSupport功能简介及原理浅析

    目录 1.LockSupport功能简介 1.1 使用wait,notify阻塞唤醒线程 1.2 使用LockSupport阻塞唤醒线程 2. LockSupport的其他特色 2.1 可以先唤醒线程 ...

  2. spring初始化源码浅析之关键类和扩展接口

    目录 1.关键接口和类 1.1.关键类之 DefaultListableBeanFactory 1.2.关键类之XmlBeanDefinitionReader 1.3.关键类之ClassPathXml ...

  3. Qt QObject

    [1]Qt的QObject 1.测试代码如下: #include<QApplication> #include<QPushButton> #include<QDebug& ...

  4. DYNAMIC_DOWNCAST、STATIC_DOWNCAST、CRuntimeClass和IsKindOf

    DYNAMIC_DOWNCAST.STATIC_DOWNCAST.CRuntimeClass和IsKindOf   DYNAMIC_DOWNCAST(class, pointer ):如果pointe ...

  5. VC 宏与预处理使用方法总结

    目录(?) C/C++ 预定义宏^ C/C++ 预定义宏用途:诊断与调试输出^ CRT 和 C 标准库中的宏^ NULL 空指针^ limits.h 整数类型常量^ float.h 浮点类型常量^ m ...

  6. Qt源码分析之QObject

    原文:http://blog.csdn.net/oowgsoo/article/details/1529284 我感觉oowgsoo兄弟写的分析相当透彻,赞! 1.试验代码: #include < ...

  7. Android 内核初识(7)RefBase、LightRefBase、sp和wp

    简介 RefBase是Android中所有对象的始祖,类似MFC中的CObject及Java中的Object对象.在Android中,RefBase结合sp和wp,实现了一套通过引用计数的方法来控制对 ...

  8. halcon基础数据类型详解

    #if defined(__CHAR_UNSIGNED__) || defined(__sgi) #define INT1 signed char /* integer, signed 1 Byte ...

  9. kubernetes基础概念知多少

    kubernetes(简称k8s)是一种用于在一组主机上运行和协同容器化应用程序的管理平台,皆在提供高可用.高扩展性和可预测性的方式来管理容器应用的生命周期.通过k8s,用户可以定义程序运行方式.部署 ...

随机推荐

  1. Servlet跳转

    方便自己查询,嫌低级的勿喷.... 在Servlet中跳转有两种: 1.客户端跳转 在Servlet中要进行客户端跳转(地址栏的地址信息将发生改变),直接使用HttpServletResponse接口 ...

  2. Java 判断一段网络资源是否存在

    package cn.ycmedia.common.utils; import java.io.InputStream; import java.net.URL; import java.net.UR ...

  3. oracle日期时间函数总结

    常常写 sql 的同学应该会接触到一些 oracle 的日期时间函数, 比如: 財务软件或者人力资源软件须要依照每年, 每季度, 每月, 甚至每一个星期来进行统计. 今天闲来没事, 特意从网上整理了一 ...

  4. mysql 加入�列,改动列,删除列。

    MySQL 加入�列,改动列,删除列 ALTER TABLE:加入�,改动,删除表的列,约束等表的定义. 查看列:desc 表名; 改动表名:alter table t_book rename to ...

  5. PHP Predefined Interfaces 预定义接口(转)

    SPL提供了6个迭代器接口: Traversable 遍历接口(检测一个类是否可以使用 foreach 进行遍历的接口) Iterator 迭代器接口(可在内部迭代自己的外部迭代器或类的接口) Ite ...

  6. linux nadianshi

    http://www.cnblogs.com/fnng/archive/2012/03/19/2407162.html

  7. linux 一些笔记内容

    #which COMMAND : 显示命令路径#whatis COMMAND : 命令出现在哪个章节#type COMMAND :显示一个命令是内部命令还是外部命令#printenv :显示系统信息 ...

  8. Android更新UI的几种方式

    之前做过一个Android采集心电图数据的程序,那才是真正的多线程,之前写的小程序:比如下载个文件,从socket接受大一点的数据流然后在ui上更新进度,我都感觉这就叫做多线程了,其实这啥都不算,用个 ...

  9. 模板-->Matrix重载运算符:+,-,x

    如果有相应的OJ题目,欢迎同学们提供相应的链接 相关链接 所有模板的快速链接 poj_2118_Firepersons,my_ac_code 简单的测试 INPUT: 1 2 3 1 3 4 3 -1 ...

  10. css3 2D变换 transform

    旋转函数rotate(),deg表示度数,transform-origin表示旋转的基点 <head> <title>无标题文档</title> <style ...