MFC浅析(4) CObject浅析
MFC CObject浅析
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浅析的更多相关文章
- 阻塞和唤醒线程——LockSupport功能简介及原理浅析
目录 1.LockSupport功能简介 1.1 使用wait,notify阻塞唤醒线程 1.2 使用LockSupport阻塞唤醒线程 2. LockSupport的其他特色 2.1 可以先唤醒线程 ...
- spring初始化源码浅析之关键类和扩展接口
目录 1.关键接口和类 1.1.关键类之 DefaultListableBeanFactory 1.2.关键类之XmlBeanDefinitionReader 1.3.关键类之ClassPathXml ...
- Qt QObject
[1]Qt的QObject 1.测试代码如下: #include<QApplication> #include<QPushButton> #include<QDebug& ...
- DYNAMIC_DOWNCAST、STATIC_DOWNCAST、CRuntimeClass和IsKindOf
DYNAMIC_DOWNCAST.STATIC_DOWNCAST.CRuntimeClass和IsKindOf DYNAMIC_DOWNCAST(class, pointer ):如果pointe ...
- VC 宏与预处理使用方法总结
目录(?) C/C++ 预定义宏^ C/C++ 预定义宏用途:诊断与调试输出^ CRT 和 C 标准库中的宏^ NULL 空指针^ limits.h 整数类型常量^ float.h 浮点类型常量^ m ...
- Qt源码分析之QObject
原文:http://blog.csdn.net/oowgsoo/article/details/1529284 我感觉oowgsoo兄弟写的分析相当透彻,赞! 1.试验代码: #include < ...
- Android 内核初识(7)RefBase、LightRefBase、sp和wp
简介 RefBase是Android中所有对象的始祖,类似MFC中的CObject及Java中的Object对象.在Android中,RefBase结合sp和wp,实现了一套通过引用计数的方法来控制对 ...
- halcon基础数据类型详解
#if defined(__CHAR_UNSIGNED__) || defined(__sgi) #define INT1 signed char /* integer, signed 1 Byte ...
- kubernetes基础概念知多少
kubernetes(简称k8s)是一种用于在一组主机上运行和协同容器化应用程序的管理平台,皆在提供高可用.高扩展性和可预测性的方式来管理容器应用的生命周期.通过k8s,用户可以定义程序运行方式.部署 ...
随机推荐
- string和stringbuffer stringbuilder的快速理解。
这三个对象都可操作字符串,区别string定义的变量除非重新赋值,否则是不可改变的.调用string的方法不会改变,但是其他两个有对象的方法可改变,比如apend的方法,后两个区别一个是线程安全不安全 ...
- [Flux] Stores
Store, in Flux which manager the state of the application. You can use EventEmiiter to listen to the ...
- String.Split()函数
我们在上次学习到了 String.Join函数(http://blog.csdn.net/zhvsby/archive/2008/11/28/3404704.aspx),当中用到了String.SPl ...
- ubuntu vim YouComlpeteMe配置
使用vundle安装时,在.vimrc中添加 Plugin 'Valloric/YouCompleteMe' 使用Bundle会安装失败原因未知 YCM编译时附加选项--system-libclang ...
- iOS 启动连续闪退保护方案
引言 “如果某个实体表现出以下任何一种特性,它就具备自主性:自我修复.自我保护.自我维护.对目标的自我控制.自我改进.” —— 凯文·凯利 iOS App 有时可能遇到启动必 crash 的绝境:每次 ...
- jquery之隐藏div
1.$("#demo").attr("style","display:none;");//隐藏div $("#demo" ...
- 优雅退出 Android 应用程序的 6 种方式
我们先来看看几种常见的退出方法(不优雅的方式) 一.容器式 建立一个全局容器,把所有的Activity存储起来,退出时循环遍历finish所有Activity import java.util.Arr ...
- 开发solr功能问题点
Criteria criteria = new Criteria(); public Criteria and(Criteria criteria) { this.criteriaChain.add( ...
- 详解Linux服务器最大tcp连接数
网络编程在tcp应用中,server事先在某个固定端口监听,client主动发起连接,经过三路握手后建立tcp连接.那么对单机,其最大并发tcp连接数是多少? 如何标识一个TCP连接在确定最大连接数之 ...
- JavaScript--声明提前
声明提前(hoist): 在正式执行程序前,都会将所有var声明的变量和function声明的函数提前到*当前作用域*的顶部集中创建. 但是,赋值留在原地. console.log(a);//unde ...