探察MFC中框架宏(RUNTIME_CLASS等)的秘密

      学mfc学到文档,视图和框架的时候,知道必须在这三个类的派生类的类声明里加上DECLARE_DYNCREATE,然后在类声明外合适的地方加上IMPLEMENT_DYNCREATE,然后文档,视图和框架,还有文档模板就可以协调工作了。查看msdn,发现类似的宏有这几对:
      DECLARE_DYNAMIC 和 IMPLEMENT_DYNAMIC
      DECLARE_DYNCREATE 和 IMPLEMENT_DYNCREATE
      DECLARE_SERIAL 和 IMPLEMENT_SERIAL
      虽然msdn里介绍了他们的作用,但对于它们为什么会起这样的作用心里却没底,于是翻了翻mfc的源代码,喜欢钻牛角尖的人可以和我一起来钻一钻。

    >RUNTIME_CLASS
      RUNTIME_CLASS宏的定义是这样的:
#define RUNTIME_CLASS(class_name)
((CRuntimeClass*)(&class_name::class##class_name))
      其中##的意思是把##两边的符号都进行宏扩展(如果它们是宏的话),然后把扩展后的内容连接在一起,中间不加空格。例如:RUNTIME_CLASS(CView)将被扩展成: (CRuntimeClass*)(&CView::classCView) 但这个classCView是什么意思?

      原来,classCView是由DECLARE_DYNAMIC(CView) 引入的一个public属性的CRuntimeClass类型的静态成员变量:
      static const AFX_DATA CRuntimeClass classCView;
      原来RUNTIME_CLASS的作用就是引用由DECLARE_DYNAMIC宏引入的静态成员变量。

     >DECLARE_DYNAMIC

      DECLARE_DYNAMIC(class_name)
     由于篇幅的原因,宏的具体定义代码就不列出来了,感兴趣的可以去看文件afx.h。该宏往类中声明了三个成员:
protected:
static CRuntimeClass* PASCAL _GetBaseClass();
public:
virtual CRuntimeClass* GetRuntimeClass() const;
static const AFX_DATA CRuntimeClass class##class_name;
     有两个成员函数,一个静态成员变量class+类名,同RUNTIME_CLASS相似,如果是DECLARE_DYNAMIC(CView)的话,这个静态成员变量将是classCView。可见这个成员变量的名称是和DECLARE_DYNAMIC的参数有关的。在下文我们把这个成员变量统统记做class##class_name。这个静态成员和两个成员函数在哪里被初始化和具体实现呢?原来是在IMPLEMENT_DYNAMIC宏里。

    >IMPLEMENT_DYNAMIC

      IMPLEMENT_DYNAMIC(class_name, base_class_name)
      查看它的宏定义,如果_AFXDLL被定义了的话,由DECLARE_DYNAMIC引入的成员的初始化和实现是这样的:
CRuntimeClass* PASCAL class_name::_GetBaseClass()
{
        return RUNTIME_CLASS(base_class_name);
}
CRuntimeClass* class_name::GetRuntimeClass() const
{
         return RUNTIME_CLASS(class_name);
}
AFX_COMDAT const AFX_DATADEF
CRuntimeClass class_name::class##class_name =
{
#class_name,
sizeof(class class_name),
0xFFFF,
NULL,
NULL,
&class_name::_GetBaseClass,
NULL
};//这是在初始化静态成员变量class##class_name。//CRuntimeClass结构的各个成员的意义可查看msdn。

    >DECLARE_DYNAMIC 
      _DECLARE_DYNAMIC(class_name)该宏的定义和DECLARE_DYNAMIC(class_name)基本一样。不同之处是静态成员
class##class_name前面没有const修饰符。

      >DECLARE_DYNCREATE
      DECLARE_DYNCREATE(class_name)
      该宏也往类中引入了DECLARE_DYNAMIC宏所引入的那三个成员。除此之外,它还另外引入了一个成员: 
            static CObject* PASCAL CreateObject();
      该宏引入的成员在IMPLEMENT_DYNCREATE里初始化和实现。 
    >IMPLEMENT_DYNCREATE(class_name, base_class_name)
     该宏自然是初始化和实现由DECLARE_DYNCREATE引入的成员了。我们看看CreateObject的实现:
CObject* PASCAL class_name::CreateObject()

return new class_name;
}
       呵,这个函数是如此简单,它就是用CObject类里重载的new操作符创建一个该类类型的对象。
    >_DECLARE_DYNCREATE(class_name)
      该宏引入了和DECLARE_DYNCREATE引入的四个成员差不多的成员。唯一的区别是该宏引入的静态成员class##class_name前面没有const修饰符。
    >DECLARE_SERIAL(class_name)
      该宏引入了和_DECLARE_DYNCREATE所引入的一样的四个成员,另外它还多了这么一句:
      AFX_API friend CArchive& AFXAPI operator>>(CArchive& ar, class_name* & pOb); 原来是把重载操作符operator>>的函数当作该类的友元。于是在操作符函数operator>>中就可以访问该类的成员了。
    >IMPLEMENT_SERIAL(class_name, base_class_name, wSchema)
      该宏初始化了成员变量:
CRuntimeClass class_name::class##class_name=
{
      #class_name,
      sizeof(class class_name),
      wSchema,
      class_name::CreateObject),
      RUNTIME_CLASS(base_class_name),
      NULL
};//在这里,class##class_name前面是没有const修饰符的。
      该宏还实现了下列函数:
CObject* PASCAL class_name::CreateObject()
{
      return new class_name;
}
CRuntimeClass* class_name::GetRuntimeClass() const
{
      return RUNTIME_CLASS(class_name);
}
CArchive& AFXAPI operator>>(CArchive& ar, class_name* &pOb)

      pOb=(class_name*)ar.ReadObject(RUNTIME_CLASS(class_name));
      return ar;
}
      该宏还声明了一个函数原型:
AFX_CLASSINIT _init_##class_name(RUNTIME_CLASS(class_name));

时间: 2024-10-22 02:30:15

探察MFC中框架宏(RUNTIME_CLASS等)的秘密的相关文章

MFC中常用类、宏、函数的简单介绍

闲来无事,整理了一下MFC中常用的类.宏.函数. 常用类 CRect:用来表示矩形的类,拥有四个成员变量:top left bottom right.分别表是左上角和右下角的坐标.可以通过以下的方法构造: CRect( int l, int t, int r, int b ); 指明四个坐标 CRect( const RECT& srcRect ); 由RECT结构构造 CRect( LPCRECT lpSrcRect ); 由RECT结构构造 CRect( POINT point, SIZE

线程-mfc中定义大量宏的问题

问题描述 mfc中定义大量宏的问题 我定义了大量的宏,在cmainframe和一个共有类(用于存放一些静态函数和变量,主要是为了线程函数中使用全局变量)中里面的一个函数中需要使用到,我直接把它们放在一个头文件中然后include,就报了error LNK2005: ""struct Zeni_MTS_Rolypon_InOut InOutRoly"" (?InOutRoly@@3UZeni_MTS_Rolypon_InOut@@A) already defined

更新MFC中的视图,跟踪.NET Framework中的事件

本文配套源码 如何更新MFC中的视图? 如何跟踪.NET Framework 中的事件? 我在 MDI 程序中打算通过 CMainFrame 中的定时器事件来更新所有的子窗口. 视图用于显示许多图表.用如下的代码只能更新当前活动窗口: GetActiveWindow()->GetActiveView()->GetDocument() 是否有其它的方法从 CMDIFrame 类中获得所有的子窗口或者所有的文档? Makarand 你的情况并不罕见.许多采集实时数据的程序需要定时更新屏幕.即使你的

谈static在MFC中的运用

学习<深入浅出MFC>有一段时间了,有一些想法就记了下来,希望与大家共勉 我们知道,C++中的static数据成员表示基于该类创建的所有对象共享该static数据成员的一份拷贝.基于这个原则,可以实现多种技术,比如说为某类创建一个计数器,用来记录任意时刻共创建了多少个此类的对象. class test{ public: test(){count++;} static int count; }; test::count=0; void main(){ cout<<test::coun

c++-mfc中 的字符插入表格中的问题

问题描述 mfc中 的字符插入表格中的问题 字符串格式:{小明|25|2011070737|}{小红|24|2011070764} 表格列名:{姓名|年龄|学号|} 想写一个函数 根据判断为一组数据 把上边两组数据插入到列名如上的中表格 解决方案 百度一下你就知道了啊啊啊啊啊啊 啊 解决方案二: 思路挺清楚,就是问题不明确,具体哪里不会?建议你先试着写个框架,说明具体有问题. 解决方案三: 建议用map,map 结构体定义为姓名和年龄, 第一个参数为学号, 生成的map默认按学号升序排列

请问mfc中如何实现工具栏按钮的自定义添加与删除

问题描述 请问mfc中如何实现工具栏按钮的自定义添加与删除 mfc单文档程序运行后,左上角有一个系统自动生成的工具栏,点击该工具栏右边的小倒三角形,会出现如下图所示的自定义添加或删除按钮的选项 现在我自己创建了一个工具栏,想实现上述功能,请问该怎么做?谢谢~~~ 解决方案 难道你的没有么?你可以试试BCGControlBar,它是CMFCxxx控件的商业完整版本.包含了和Office一样的自定义对话框,不需要写任何代码. 解决方案二: 需要用MFC等提供的框架,才会有,它帮你实现了.你自己创建的

教程-MFC中界面美化与皮肤制作

问题描述 MFC中界面美化与皮肤制作 现在只会用MFC编写一些小程序,但界面一直都是默认的那种,顶多就能换个对话框背景,想要作一些界面比较好看的程序,请大神给一些教程,之前在网上查的用skinsharp和SHEditor,但也不是太懂 解决方案 Windows UI 框架 解决方案二: MFC从入门到精通 你学这个之前必须得了解计算机系统消息的处理机制 会很有帮助

mfc-新手遇到的关于MFC中的一些问题.

问题描述 新手遇到的关于MFC中的一些问题. 这种界面可以用什么模式编写?SDI/MDI还是基于对话框?每种模式都可以建立标签吗? 解决方案 MFC向导sdi,选择资源管理器样式,或者在高级设置中添加一个面板,都可以. 解决方案二: 效果图 解决方案三: Sdi,然后切割窗口,左边窗口CFormView,放各种控件,右边CView等,画图. 解决方案四: SDI工程,基类选择CView/CFormView都可以,利用CSplitterWnd拆分视图框架窗口,左边那个View可以用CFormVie

c++-mfc中关于未声明的标识符的一个问题

问题描述 mfc中关于未声明的标识符的一个问题 请问各位这个问题怎么解决? 解决方案 // MFCApplication1.cpp : Defines the class behaviors for the application.//#include ""stdafx.h""#include ""afxwinappex.h""#include ""afxdialogex.h""#inc