`
congfeng02
  • 浏览: 195330 次
  • 性别: Icon_minigender_1
  • 来自: 北京
文章分类
社区版块
存档分类
最新评论

MFC CObject浅析

阅读更多

FMD开发文集 -- MFC CObject浅析
作者:冯明德

CObject是大部分的MFC类的基类 为了完成MFC类的判断、动态生成、序列化等特殊功能,CObject中添加了特定的处理。 为了进一步增强对MFC类对象的理解,在此对CObject源码及相关宏定义进行分析。 (所附代码并非原始代码,为说明问题而作了删减。) 主要介绍了以下几方面内容:

1.CObject简要声明
2.CRuntimeClass结构
3.RUNTIME_CLASS
4.DYNAMIC支持
5.DYNCREATE支持
6.SERIAL支持

一.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();

};
在此声明中很多都是纯虚函数,定义的一个一般对象的"界面"

二.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
};

三.RUNTIME_CLASS

RUNTIME_CLASS(class_name)用于返回指向运行时类信息结构的指针,定义如下:
#define RUNTIME_CLASS(class_name) ((CRuntimeClass*)(&class_name::class##class_name))
四.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判断类类型了。

五.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)
六.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);以实现类数据的保存及建立后读入。 从而实现类的保存,及读入动态建立。
分享到:
评论

相关推荐

    网上资料整理—MFC浅析(pdf)

    网上资料整理。简要解析了MFC中几类关键问题的机理、过程...MFC浅析(4) CObject浅析 MFC浅析(5) 命令更新机制 MFC浅析(6) 对话框数据交换及验证 MFC浅析(7) CWnd类虚函数的调用时机、缺省实现 MFC浅析(8) CArchive 原理

    MFC编程书籍---MFC浅析

    根据网上资料进行编辑、代码着色整理而成,对使用MFC编程...文档视图结构缺省的命令处理、操作流程 、处理流程、CObject浅析 、命令更新机制、对话框数据交换及验证 、CWnd类虚函数的调用时机、缺省实现、CArchive 原理

    MFC教程CObject类.pdf

    MFC教程CObject类.pdf

    MFC中CObject1

    摘要视图订阅文档CSDN日报20170328——《你看那个人他像一条狗》 CSDN 知识小饭桌:Python 进阶 Q & A 程序员3月书讯MFC中CObje

    MFC教程.rar(MFC的状态/MFC教程_ CObject类/)

    MFC的状态/MFC教程_ CObject类/

    MFC中文教程 李久进

    mfc3CObject类.doc mfc4消息映射的实现.doc mfc5MFC对象创建.doc mfc6应用程序的退出.doc mfc7MFC的DLL.doc mfc8MFC的进程和线程.doc mfc9MFC的状态.doc mfc10内存的分配方式和调试机制.doc mfc11MFC下的文件类.doc ...

    MFC中使用自定义CObject派生类重载“=”“==”操作符 errorC2248 errorC2678

    类的话就让他派生自CObject吧,然后接下来会遇到两个问题,我在这里整理一下我在网上寻求的解决方法,文中所写是否是问题的标准解决办法,有待高手指教,我只是发此文章帮助和我一样遇到问题的人,和记录一下自己的...

    《对话框》之《CObject类》

    CObject是大多数MFC类的根类或基类。CObject类有很多有用的特性:对运行时类信息的支持,对动态创建的支持,对串行化的支持,对象诊断输出,等等。MFC从CObject派生出许多类,具备其中的一个或者多个特性。程序员也...

    MFC完全自学资料,基础教程+图+详细说明,从此不怕MFC

    3_CObject类, 4_消息映射的实现, 5_MFC对象的创建, 6_应用程序的退出, 7_MFC的DLL, 8_MFC的进程和线程, 9_MFC的状态, 10_内存分配方式和调试机制, 11_MFC下的文件类, 12_对话框和对话框类CDialog, 13_MFC...

    编程常用MFC类说明

    CObject类:CObject是MFC的根类,绝大多数MFC类是其派生的,CObject 实现了一些重要的特性,包括动态类信息、动态创建、对象序列化、对程序调试的支持,等等。CObject定义了一个CRuntimeClass类型的静态成员变量:...

    深入解析MFC

    这是一本填补“使用向导”类的VisualC++书籍、产品文档以及MFC源代码之间空隙的MFC书籍。本书是了解MFC内幕的向导,提供了关于那些没有文档记录的MFC类、实用函数和数据成员的独一无二并且透彻的信息,介绍了有用的...

    MFC中使用自定义CObject派生类重载= ==操作符 解决errorC2248 errorC2678

    类的话就让他派生自CObject吧,然后接下来会遇到两个问题,我在这里整理一下我在网上寻求的解决方法,文中所写是否是问题的标准解决办法,有待高手指教,我只是发此文章帮助和我一样遇到问题的人,和记录一下自己的...

    MFC 教程.rar 网页版

    │ MFC教程_ CObject类.htm │ MFC教程_ MFC下的文件类.htm │ MFC教程_ MFC和Win32.htm │ MFC教程_ MFC对象的创建.htm │ MFC教程_ MFC工具条和状态栏.htm │ MFC教程_ MFC的DLL.htm │ MFC教程_ MFC的进程和线程....

    MFC深入浅出带目录完整版(李久进chm版)

    第三章,讨论CObject的特性及其实现,包括动态类信息、动态创建、序列化的实现等内容。 第四章,讨论MFC的消息映射机制,分析MFC对各类消息的处理,例如对Windows消息、控制通知消息、命令消息、状态更新消息、反射...

    mfc课件 第3章 CObject派生类.ppt

    mfc课程教学课件。 该部分是mfc课程的第三部分。

    MFC深入浅出(完整版)

    第三章 CObject类 第四章 消息映射的实现 第五章 MFC对象的创建 第六章 应用程序的退出 第七章 MFC的DLL 第八章 MFC的进程和线程 第九章 MFC的状态 第十章 内存分配方式和调试机制 第十一章 MFC下的文件类 第十二章 ...

    学习MFC的入门资料——MFC教程

    MFC教程(概述,MFC和Win32, CObject类,消息映射的实现,MFC对象的创建,应用程序的退出,MFC的DLL,MFC的进程和线程,MFC的状态等)

    MFC的PC端系统

    CObject是MFC的根类,绝大多数MFC类是其派生的,包括CCmdTarget。CObject 实现了一些重要的特性,包括动态类信息、动态创建、对象序列化、对程序调试的支持,等等。所有从CObject派生的类都将具备或者可以具备...

    mfc教程(word版)

    第1章 MFC概述 1 1.1 MFC是一个编程框架 1 1.1.1 封装 1 1.1.2 继承 2 1.1.3 虚拟函数和动态约束 2 1.1.4 MFC的宏观框架体系 2 1.2 MDI应用程序的构成 3 1.2.1 构成应用程序的对象 3 1.2.2 构成应用程序的对象之间的...

    MFC深入浅出(李久进)图全

    第三章,讨论CObject的特性及其实现,包括动态类信息、动态创建、序列化的实现等内容。 第四章,讨论MFC的消息映射机制,分析MFC对各类消息的处理,例如对Windows消息、控制通知消息、命令消息、状态更新消息、反射...

Global site tag (gtag.js) - Google Analytics