注册通行证 用户名 密码
  • 文章投稿
  • 博客
  • 论坛
  • 设为首页
  • 加入收藏
jztop.com网络技术
  • 首页
  • | iT新闻
  • | 操作系统
  • | 组网建网
  • | 网络安全
  • | 程序开发
  • | 办公一族
  • | 工具软件
  • | 网页制作
  • | 多媒体制作
  • | 网吧技术
  • | 服务器
  • | 专题教程
Vista | 软件评测 | 系统备份 | 优化 | 进程 | 聊天 | 病毒 | Linux | 黑客 | 防火墙 | 数据库 | Web开发 | Java | Word | 游戏 | 32位开发 | 移动开发
当前位置:首页 > 程序开发 > 32位开发 > C/C++ 内容正文:MFC六大关键技术剖析之动态创建

MFC六大关键技术剖析之动态创建

发布时间:2006-05-05 19:01:16 来源:网易社区 网友评论 0 条

  动态创建就是运行时创建指定类的对象,在MFC中大量使用。如框架窗口对象、视对象,还有文档对象都需要由文档模板类对象来动态的创建。我觉得这是每个MFC的学习者很希望理解的问题。

   初次接触MFC的时候,很容易有这样的迷惘。MFC的几大类不用我们设计也就罢了,但最疑惑的是不用我们实例化对象。本来最直观的理解就是,我们需要框架的时候,亲手写上CFrameWnd myFrame;需要视的时候,亲自打上CView myView;……

   但MFC不给我们这个机会,致使我们错觉窗口没有实例化就弹出来了!就象画了张电视机的电路图就可以看电视一样令人难以置信。但大伙想了一下,可能会一拍脑门,认为简单不过:MFC自动帮我们完成CView myView之流的代码不就行了么!!!其实不然,写MFC程序的时候,我们几乎要对每个大类进行派生改写。换句话说,MFC并不知道我们打算怎样去改写这些类,当然也不打算全部为我们“静态”创建这些类了。即使静态了创建这些类也没有用,因为我们从来也不会直接利用这些类的实例干什么事情。我们只知道,想做什么事情就往各大类里塞,不管什么变量、方法照塞,塞完之后,我们似乎并未实例化对象,程序就可以运行!

   要做到把自己的类交给MFC,MFC就用同一样的方法,把不同的类一一准确创建,我们要做些什么事情呢?同样地,我们要建立链表,记录各类的关键信息,在动态创建的时候找出这些信息,就象上一节RTTI那样!我们可以设计一个类:

struct CRuntimeClass{
  LPCSTR m_lpszClassName; //类名指针
  CObject* (PASCAL *m_pfnCreateObject)(); //创建对象的函数的指针
  CRuntimeClass* m_pBaseClass; //讲RTTI时介绍过
  CRuntimeClass* m_pNextClass; //指向链表的下一个元素(许多朋友说上一节讲RTTI时并没有用到这个指针,我原本以为这样更好理解一些,因为没有这个指针,这个链表是无法连起来,而m_pBaseClass仅仅是向基类走,在MFC的树型层次结构中m_pBaseClass是不能遍历的)

  CObject* CreateObject(); //创建对象
  static CRuntimeClass* PASCAL Load(); //遍历整个类型链表,返回符合动态创建的对象。
  static CRuntimeClass* pFirstClass; //类型链表的头指针
};

   一下子往结构里面塞了那么多的东西,大家可以觉得有点头晕。至于CObject* (PASCAL *m_pfnCreateObject)();,这定义函数指针的方法,大家可能有点陌生。函数指针在C++书籍里一般被定为选学章节,但MFC还是经常用到此类的函数,比如我们所熟悉的回调函数。简单地说m_pfnCreateObject即是保存了一个函数的地址,它将会创建一个对象。即是说,以后,m_pfnCreateObject指向不同的函数,我们就会创建不同类型的对象。

   有函数指针,我们要实现一个与原定义参数及返回值都相同一个函数,在MFC中定义为:

static CObject* PASCAL CreateObject(){return new XXX};//XXX为类名。类名不同,我们就创建不同的对象。

   由此,我们可以如下构造CRuntimeClass到链表:

CRuntimeClass classXXX={
  类名,
  ……,
  XXX::CreateObject(), //m_pfnCreateObject指向的函数
  RUNTIME_CLASS(基类名) // RUNTIME_CLASS宏可以返回CRuntimeClass对象指针。
  NULL //m_pNextClass暂时为空,最后会我们再设法让它指向旧链表表头。
};

   这样,我们用函数指针m_pfnCreateObject(指向CreateObject函数),就随时可new新对象了。并且大家留意到,我们在设计CRuntimeClass类对时候,只有类名(和基类名)的不同(我们用XXX代替的地方),其它的地方一样,这正是我们想要的,因为我们动态创建也象RTTI那样用到两个宏,只要传入类名和基类作宏参数,就可以满足条件。

   即是说,我们类说明中使用DECLARE_DYNCREATE(CLASSNMAE)宏和在类的实现文件中使用IMPLEMENT_DYNCREATE(CLASSNAME,BASECLASS)宏来为我们加入链表,至于这两个宏怎么为我们建立一个链表,我们自己可以玩玩文字代换的游戏,在此不一一累赘。但要说明的一点就是:动态创建宏xxx_DYNCREATE包含了RTTI宏,即是说, xxx_DYNCREATE是xxx_DYNAMIC的“增强版”。

   到此,我们有必要了解一下上节课没有明讲的m_pNextClass指针。因为MFC层次结构是树状的,并不是直线的。如果我们只有一个m_pBaseClass指针,它只会沿着基类上去,会漏掉其它分支。在动态创建时,必需要检查整个链表,看有多少个要动态创建的对象,即是说要从表头(pFirstClass)开始一直遍历到表尾(m_pNextClass=NULL),不能漏掉一个CRuntimeClass对象。

   所以每当有一个新的链表元素要加入链表的时候,我们要做的就是使新的链表元素成为表头,并且m_pNextClass指向原来链表的表头,即像下面那样(当然,这些不需要我们操心,是RTTI宏帮助我们完成的):

pNewClass->m_pNextClass=CRuntimeClass::pFirstClass;//新元素的m_pNextClass指针指向想加入的链表的表头。

CRuntimeClass::pFirstClass=pNewClass;//链表的头指针指向刚插入的新元素。


   好了,有了上面的链表,我们就可以分析动态创建了。

  有一了张有类名,函数指针,动态创建函数的链表,我们就可以知道应该按什么步骤去动态创建了:

   1、获得一要动态创建的类的类名(假设为A)。

   2、将A跟链表里面每个元素的m_lpszClassName指向的类名作比较。

   3、若找到跟A相同的类名就返回A所属的CRuntimeClass元素的指针。

   4、判断m_pfnCreateObject是否有指向创建函数,有则创建对象,并返回该对象。

   代码演示如下(以下两个函数都是CRuntimeClass类函数):

   ///////////////以下为根据类名从表头向表尾查找所属的CRuntimeClass对象////////////

CRuntimeClass* PASCAL CRuntimeClass::Load()
{
  char szClassXXX[64];
  CRuntimeClass* pClass;
  cin>>szClassXXX; //假定这是我们希望动态创建的类名
  for(pClass=pFirstClass;pClass!=NULL;pClass=pClass->m_pNextClass)
  {
   if(strcmp(szClassXXX,pClass->m_lpszClassName)==0)
    return pClass;
  }
  return NULL
}

///////////根据CRuntimeClass创建对象///////////

CObject* CRuntimeClass::CreateObject()
{
  if(m_pfnCreateObject==NULL) return NULL;
  CObject *pObject;
  pObject=(* m_pfnCreateObject)(); //函数指针调用
  return pObject;
}

   有了上面两个函数,我们在程序执行的时候调用,就可以动态创建对象了。

   我们还可以更简单地实现动态创建,大家注意到,就是在我们的程序类里面有一个RUNTIME_CLASS(class_name)宏,这个宏在MFC里定义为:

RUNTIME_CLASS(class_name) ((CRuntimeClass*)(&class_name::class##class_name))

   作用就是得到类的RunTime信息,即返回class_name所属CRuntimeClass的对象。在我们的应用程序员类(CMyWinApp)的InitInstance()函数下面的CSingleDocTemplate函数中,有:

RUNTIME_CLASS(CMyDoc),

RUNTIME_CLASS(CMainFrame), // main SDI frame window

RUNTIME_CLASS(CMyView)

   构造文档模板的时候就用这个宏得到文档、框架和视的RunTime信息。有了RunTime信息,我们只要一条语句就可以动态创建了,如:

classMyView->CreateObject(); //对象直接调用用CRuntimeClass本身的CreateObject()

   现在,细心的朋友已经能清楚动态创建需要的步骤:

   1、定义一个不带参数的构造函数(默认构造函数);因为我们是用CreateObject()动态创建,它只有一条语句就是return new XXX,不带任何参数。所以我们要有一个无参构造函数。

   2、类说明中使用DECLARE_DYNCREATE(CLASSNMAE)宏;和在类的实现文件中使用IMPLEMENT_DYNCREATE(CLASSNAME,BASECLASS)宏;这个宏完成构造CRuntimeClass对象,并加入到链表中。

   3、使用时先通过宏RUNTIME_CLASS得到类的RunTime信息,然后使用CRuntimeClass的成员函数CreateObject创建一个该类的实例。

   4、CObject* pObject = pRuntimeClass->CreateObject();//完成动态创建。
相关文章
  • VC++动态链接库编程之MFC扩展 DLL
  • 在MFC下如何定义全局变量和全局函数
  • 经典与现代的结合:在MFC中集成RAD .NET框架
  • 向任何应用程序发送击键,从MFC应用程序中调用.NET及其
  • MFC中几个有用的字符串操作函数
【评论】【收藏本文】【打印】【关闭】
上一篇文章:c++类静态数据成员与类静态成员函数
下一篇文章:C++类对象的复制-拷贝构造函数(深拷贝,浅拷贝)
讨论区
查看
已有 0 位对此新闻感兴趣的网友发表了看法
匿名发表
注册通行证 登陆
图文阅读推荐
推荐阅讯
  • C/C++中枚举类型(enum)的入门教程
  • c++类静态数据成员与类静态成员函数
  • C++ Builder构建算二十四点小游戏
  • C/C++中利用空指针(NULL),提高程序运行效率
  • 用托管C++编写自定义Web组合控件
  • 实例解析C++/CLI的输入与输出
  • 新手入门:C/C++中的结构体
  • C++中要求(或禁止)对象产生于heap中
  • 基于VC的WinSock网络编程实用宝典
  • Visual C++ 入门精解-WINDOWS应用程序设计
阅读排行
  • 1.Borland 发布C++ Builder 2006 RAD 环境
  • 2.C/C++程序员应聘常见面试题深入剖析
  • 3.Visual C++常用数据类型转换详解
  • 4.C++中的 static 关键字
  • 5.利用VC++实现局域网实时视频传输
  • 6.浅谈C/C++内存泄漏及其检测工具
  • 7.英国投票否决C++/CLI 微软强攻ISO标准受挫
  • 8.VC++下用MSComm控件实现串口通讯
  • 9.伪随机数生成及在VC++中的实现
  • 10.VC++编程实现对波形数据的频谱分析
专题教程
  • 大话G游 专题:手机病毒揭密
  • ARP攻击防范与解决方案 路由故障处理手册
  • Picasa中文版_Picasa教程 专题:清除流氓软件
  • Firefox专题 seo搜索引擎优化专区
  • 重装Windows必知的事情 装机之必备软件大行动
病毒专杀栏
  • 杀毒软件反被病毒杀 连"救命"都不能喊
  • 金山ARP防火墙
  • 还原卡神话破灭“机器狗”病毒来势汹汹
  • cctv经济半小时:你的手机现在安全吗?
  • 新挂马方式开始流行 ARP挂马称雄局域网
  • 木马和病毒清除的通用解法
  • IP地址不再冲突 查找ARP攻击者元凶
  • 教你几招识别和防御Web网页木马
  • 分析:封杀BT只是暂时的止痛药
  • QQ爆危险漏洞,“QQ游戏邀请大盗”邀请你玩病
关于我们 | 诚聘英才 | 联系我们 | 版权声明 | 网站大事 | 网站地图 | 意见建议
CopyRight 2005-2007 Jztop.Com 版权所有 未经许可 请勿转载