一、资源

共有三个和资源有关的文件:资源头文件resource.h、资源描述文件resource.rc和存放在res文件夹下的具体的资源如图片等。

资源头文件中全部是宏定义,应用程序需要为每个资源都定义一个标识符用整数来表示的编号,这些编号就叫做资源标识符(提高了程序的可读性)。

资源描述文件中,是为了描述资源的外观进行定义的一些语句,如菜单资源。

res文件夹下是图片等外部资源。

总之,简单的资源如控件,这类资源的外观和功能只取决于c++代码,所以不需要资源描述文件,更不需要res文件夹,只需要在资源头文件中定义标识符即可。

对于复杂的资源如菜单资源,则需要在资源描述文件中定义好外观,在资源头文件中定义好标识符,但并不需要在res文件夹中有对于的图片等外部文件。

对于位图、图标和鼠标光标等图形或其他的视频数据,需要外部单独文件的,则需要在资源头文件中定义标识符,需要在资源描述文件中说明资源的名称和外部文件的存储位置,需要在res文件夹中存放外部文件。

最终这些定义的资源,包括res文件夹下存放的外部文件会被链接器连同obj文件链接成一个整体exe文件。

二、运行时动态创建对象---利用类信息表CRuntimeClass结构体创建对象

看到标题,大部分会说“运行时创建对象”那不是小儿科,就这样(vb.net语言描述):

Dim newButton As Button = New Button()

newButton.Name = "Button1"

这的确是在运行时创建了一个按钮。不过若需按照用户要求创建按钮、复选框或者单选框怎么办,好像也好办:

Dim newControl As Control

Select Case userSelection     //userSelection 是个字符串

Case "按钮"   newControl = New Button()

Case "复选框"  newControl = New CheckBox()

....

End Select

如果用户需要的是Windows.Forms里面的数十种控件,那么你的Select语句也要写数十行吗?我当然不是想要做这种刁难的用户,但是需求总是多种多样的,若有一种方法能够在运行时任意指定对象的创建类型,甚至是用表示类型的名字的字符串创建所需的对象,该有多么方便。.net Framwork的反射机制给我们带来了解决问题的方法。MFC其实是更早的原型(虽然有缺陷),就是类信息表。

二、动态创建对象:类信息表(或称类信息结构体)

(1)MFC中CRuntimeClass结构体的定义大致如下:

struct CRuntimeClass

{

  LPCSTR m_LpszClassName;  //类名字

  CObject*(PASCAL *m_pfnCreateObject)();  //建立类的工厂函数的指针

  CObject* CreateObject();  //工厂函数原型的声明,这个无意义,暂时不要管他,因为工厂函数定义都是在这个结构体的外部,是全局的或其他类中的函数

  CRuntimeClass* m_pBaseClass;  //基类信息表(一个节点)指针

  CRuntimeClass* m_pNextClass;  //下一个类信息表(一个节点)指针

  ............  //其他的诸如,所描述的对象所占内存大小,版本号等信息

}

实际上一个MFC应用程序有一个全局的总表,便于管理,结构图如下:

这样,根据类名来创建一个对象时,就只需要遍历这个总表找到对应的创建对象的工厂函数即可。

(2)使用时,实际上只需要用到三个宏即可(CRuntimeClass(一个参数),IMPLEMENT_DYNCREATE(两个参数)),RUNTIME_CLASS(一个参数)):

其中,前两个宏需要我们在对于的头文件和cpp文件中添加,而最后一个宏不用管他,我们直接使用就行了。

首先是类信息表(我叫类信息结构)的创建宏(用到两个宏),这个宏完成了类信息结构体的声明和相关成员数据的填充工作(说白了,这个宏给一个类加入了一个含有数据的类信息结构体的实例,并且这个实例是属于这个类的静态成员变量)。

举例如下:

//CMainFrame类的声明

class CMainFrame:public CFrameWnd

{

protected:

  CMainFrame();

  DECLARE_DYNCREATE(CMainFrame)  //完成类信息结构体的声明

  ........................

}

//CMainFrame类的实现

CMainFrame::CMainFrame(){.........}

IMPLEMENT_DYNCREATE(CMainFrame,CFrameWnd)  //完成类信息结构体数据成员的填充,需要本类和基类名称。 比如这个宏中会出现:return new CMainFrame;字样,所以创建了对象。

在main函数中使用时,直接用获取“一个类信息结构体对象”的指针的全局宏RUNTIME_CLASS(一个参数)即可,实际上RUNTIME_CLASS内容就是获取类的静态成员变量“类信息结构”。形如下面:

CRuntimeClass*(&CMainFrame::类信息结构成员)。只需要记住这个宏返回一个类信息表的指针(或者说一个元数据的指针)即可。

举例如下:

void main()

{

  //动态创建对象,通过类型信息表的成员“建立类的工厂函数的指针”来调用工厂函数以便动态创建对象。

  CObject* m=RUNTIME_CLASS(CMainFrame)->m_pfnCreateObject();  //本质上是创建了一个CMainFrame对象

  CMainFrame objFrm=(CMainFrame*)m;

  //下面就可以调用objFrm对象的成员函数了。

}

最新文章

  1. WPF之Binding
  2. MathType 公式后的空格问题
  3. 解决debian中脚本无法使用source的问题
  4. checkbox radio select绑定
  5. MongoDB (十一) MongoDB 排序文档
  6. 无法找到脚本引擎Jscript解决
  7. JAVA并发实现三(线程的挂起和恢复)
  8. Jquery结合datagrid框架
  9. Struts2---OGNL表达式和EL表达式
  10. java 基础语法 1
  11. 【javaFX学习】(二) 控件手册
  12. 在本地没有安装Oracle的情况下,使用plsql远程连接数据库
  13. Spring使用ajax异步上传文件
  14. 三、checkedListBoxControl
  15. SQL Server 2012安装step by step
  16. Jenkins 对项目持续集成的配置之二 API接口自动化 Ant+Jmeter
  17. java 解析json字符串
  18. UVa 10450 - World Cup Noise
  19. js判断display隐藏显示
  20. java对象转json对象

热门文章

  1. webAPI 405
  2. Atitit swt 4.3 4.4 4.5 新特性java attilax总结
  3. Weka关联规则分析
  4. POJ 1654 area 解题
  5. 在Servlet的GenericServlet类中为什么有两个init()方法
  6. 让子元素在父元素中水平居中align-items
  7. Rancher探秘一:初识Rancher
  8. Java泛型的应用
  9. linux关机前同步数据(sync)
  10. 圆环自带动画进度条ColorfulRingProgressView