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_CLASSRUNTIME_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中CObject】是MFC框架的基础,CObject类是MFC中许多其他类的基类,它提供了一系列核心功能,使得程序员能够方便地实现对象的动态创建、运行时类信息处理、序列化以及对象诊断。下面将详细介绍CObject类的关键...
根据网上资料进行编辑、代码着色整理而成,对使用MFC编程...文档视图结构缺省的命令处理、操作流程 、处理流程、CObject浅析 、命令更新机制、对话框数据交换及验证 、CWnd类虚函数的调用时机、缺省实现、CArchive 原理
CObject类是MFC(Microsoft Foundation Classes)框架中的基础类,它是大多数MFC类的基类,提供了诸如运行时类信息、动态创建、序列化、对象诊断等关键功能。MFC是一个面向对象的C++库,它简化了Windows应用程序的...
这个压缩包“MFC教程.rar”包含了对MFC中的一个重要基础类——`CObject`的讲解,以及关于MFC状态管理的相关教程。在这个教程中,我们可以深入理解MFC如何构建对象模型,并学习`CObject`类在其中扮演的角色。 `...
### MFC浅析知识点 #### 一、文档视图结构中的缺省命令处理 文档视图结构是MFC中一种非常重要的设计模式,它通过将应用程序的数据与用户界面分离来提高代码的可维护性和可扩展性。在此模式下,很多命令ID都有其缺省...
mfc3CObject类.doc mfc4消息映射的实现.doc mfc5MFC对象创建.doc mfc6应用程序的退出.doc mfc7MFC的DLL.doc mfc8MFC的进程和线程.doc mfc9MFC的状态.doc mfc10内存的分配方式和调试机制.doc mfc11MFC下的文件类.doc ...
类的话就让他派生自CObject吧,然后接下来会遇到两个问题,我在这里整理一下我在网上寻求的解决方法,文中所写是否是问题的标准解决办法,有待高手指教,我只是发此文章帮助和我一样遇到问题的人,和记录一下自己的...
CObject类是MFC框架中的基础类,它是大多数MFC类的根源,提供了许多关键功能。CObject类的特性包括对运行时类信息的支持、动态创建的支持、串行化以及对象诊断输出等。 1. 运行时类信息的支持: CObject类提供了一...
5. **串行化(Serialization)**:MFC通过CObject类的派生类支持数据的串行化,允许数据保存到磁盘并在需要时恢复,实现了持久化存储。 **深入MFC的使用** 1. **创建MFC应用程序**:在Visual Studio中,可以使用...
1. **基础类**:如CObject,它是所有MFC类的基类,提供了对象的序列化、动态类型识别等基础功能。CString类则是处理字符串的重要工具。 2. **应用框架**:MFC通过CWinApp类提供了应用程序框架,负责应用程序的初始...
3. **结构风格**:MFC使用CObject作为所有类的基类,通过继承和多态性来实现对象的结构化。CObject提供了基本的调试支持和动态类型信息。MFC还引入了一些特定的结构,如CString,用于处理字符串,和CArray、CList、...
1. **基础类**:如CObject,它是所有MFC类的基类,提供了动态类型信息和序列化功能。 2. **文档/视图架构**:这是MFC设计的一个核心概念,用于分离数据(文档)和用户界面(视图)。文档类存储数据,视图类负责数据...
1. **基础类**:如CObject,它是所有MFC类的基类,提供了对象的序列化、动态类型查询等功能。CString类则提供了一种方便的字符串操作方式。 2. **文档/视图架构**:这是MFC的核心设计模式,用于分离数据(文档类,...
CObject是所有MFC类的基类,它提供了序列化、运行时类信息访问和动态对象创建等特性。异常处理在MFC中是处理错误和异常情况的机制,包括异常宏变化、捕获异常、异常转换和检查异常内容等。 ### 文件操作 MFC中的...
3_CObject类, 4_消息映射的实现, 5_MFC对象的创建, 6_应用程序的退出, 7_MFC的DLL, 8_MFC的进程和线程, 9_MFC的状态, 10_内存分配方式和调试机制, 11_MFC下的文件类, 12_对话框和对话框类CDialog, 13_MFC...
7. **资源处理**:MFC通过CRuntimeClass、CObject和CResourceTemplate等类提供了对资源(如菜单、对话框、图标和字符串)的管理。开发者可以通过资源编辑器创建和修改资源,然后在程序中使用。 8. **国际化和本地化...
### MFC官方开发手册 #### MFC桌面应用程序与概念 MFC(Microsoft Foundation Classes)作为Microsoft为简化Windows程序开发而提供的类库,在Windows平台上有着广泛的应用。本手册旨在提供最新的MFC开发指南,帮助...
1. **核心类**:包括CObject基类,它是所有MFC类的祖先,提供了动态类型识别和序列化等功能。其他如CWinApp,它是每个MFC应用程序的基础,管理应用程序的生命周期;CWnd,是所有窗口类的基类,负责窗口的创建、消息...
- `CObject`是MFC类层次结构中的根类,几乎所有其他MFC类都直接或间接地继承自它。 - 它提供了序列化支持、运行时类型信息等基础功能。 - 在实际编程中,我们通常不会直接使用`CObject`类,而是使用它的子类。 2...
3. **CObject类**:CObject是所有MFC类的基类,它提供了序列化、诊断输出和动态类型检查等功能。通过继承CObject,你可以为自定义类添加这些特性。 4. **消息映射的实现**:MFC使用消息映射机制将窗口消息与函数...