MFC单文档程序架构解析

这里我以科院杨老师的单文档程序来分析一下MFC单文档的程序架构,纯属个人见解,不当之处烦请指教!

首先我们了解到的是

图(一)

theApp 是唯一一个在程序形成的时候就存在的全局变量,它属于CstockAppApp类,而CstockAppApp 继承于CwinApp类,我们看一下MSDN中CwinApp的继承关系如下:

图(二)

从继承关系当中,我们发现theApp是作为程序的实体而存在的,是单文档程序的核心。

首先分析一下的是CsockAppApp这个类,这里面有一个重要的函数

BOOL CStockAppApp::InitInstance()这个函数,包含了单文档程序中重要的信息,特别要提到的是一下的一段代码:

CSingleDocTemplate* pDocTemplate;

pDocTemplate = new CSingleDocTemplate(

IDR_MAINFRAME,

RUNTIME_CLASS(CStockAppDoc),

RUNTIME_CLASS(CMainFrame),       // main SDI frame windon

RUNTIME_CLASS(CStockAppView));

这里体现了几个重要的思想,第一动态创建和动态附加的一种思想,RUNTIME_CLASS是一个宏定义,这里不做展开,可以理解为最后的函数返回了实现了如下的语句

Retrun new CmainFramw 的功能,这里将三个不同的类对象附加到CsingleDocTemplate对象中,这样使得整个程序得以整体成立。

接下来对剩下的三个类对象做分析

1. CmainFrame类

CmianFrame类作为程序的框架类,其起到了一种容器的作用,在这个容器当中可以装在多个视图,菜单,工具等。要注意的有几点,第一,因为mianfram是没有视图的,因此如果在mainframe相应Onpaint消息,自然是可以响应这个消息的,但是你会惊奇的发现,在你的绘制当中如何也不会出现你预想出现的绘制内容,为什么呢? 嘿嘿,可以想象一下你在一个word程序里面,当你关闭了所有的白板(视图)的时候,你会发现你已经无法在编写文字,道理是一样的,在mainframe里面进行绘制,程序是没有问题,但是绘制的内容是在灰色上面,windows不予显示的。

第二点你可以发现在对菜单,和工具条的单击消息进行相应的时候你可以将消息响应函数添加到cmainframe也没有将消息响应函数调价到cview当中,但是你会惊奇的发现,两者只在一个地方相应的时候,消息响应函数没有问题都能正确的执行,但是如果同时对一个按钮或者菜单进行单击消息响应的时候,你会发现windows会执行的Cview里面的消息响应函数。分析两者之间的关系,首先Cmianframe的继承关系如下:

而对Cview类而言其继承类关系如下图所示:

可以看出两个类之间并没有直接的关系,但是我们通过程序的运行可以知道消息的响应是以Cview中为主的,可以认为存在一种类似于多态的关系。

这里具体的消息路径如下:

在SDI(单文档)界面中,菜单响应遵循这样一个顺序:菜单消息先由CMainFrame类接收,CMainFrame并不直接在内部寻找对应的相应函数,而是到CView类寻找。如果CView类有该消息的响应函数,那么就直接调用CView类中的响应函数,否则,转到CDoc类寻找,如果CDoc类中存在该消息的响应函数,那么就直接调用CDoc类中的响应函数,否则,返回到CMainFrame类寻找。如果CMainFrame类中也没有,返回到CApp类中寻找。如果在CApp类中也没有找到,表示没有该菜单的响应函数。

同时CmainFrame作为整个程序的框架,它提供了程序运行的基础环境,这里再强调介绍一下两点

1. 在CmainFrame中访问Cview对象和Cdoc对象

要访问这两个对象可以使用全局函数GetActiveDocument()和GetActiveView()这样可以获得Doc对象和View对象的实体了

2. 在CmainFrame中调用Cview对象更新窗口,这里使用方法如下:GetActiveView()->Invalidate(FALSE);// 这一句会是cview调用OnDraw消息响应函数

GetActiveView()->UpdateWindow();//这句可以加上也可以不加,暂时没有发现不加会出现什么问题。

2.Cview类

Cview类作为视图类,其可以理解为一张画布,在这张画布上可以画图,也可以画控件,其重要的函数有OnDraw这是主要绘图出现的地方。在View中要实现重画的时候可以按如下方式实现调用:

void CTestView::OnChangeRect()

{

// Change Rectangle size.

m_rcBox = CRect(20, 20, 210, 210);

// Invalidate window so entire client area

// is redrawn when UpdateWindow is called.

Invalidate();

// Update Window to cause View to redraw.

UpdateWindow();

}

// On Draw function draws the rectangle.

void CTestView::OnDraw(CDC* pDC)

{

//  ..  Other draw code here.

pDC->Draw3dRect(m_rcBox, 0x00FF0000, 0x0000FF00);

}

在Cview内中要获取到CmianFrame可以使用下面的函数:

AfxGetMainWnd()

3.doc类对象。

个人觉得单文档视图的设计模式很接近MVC的设计模式,Model层可以粗略的认为是在Doc这个类对象里面,在单文档视图当中Doc对象充当的也是数据存储的类,可见确实是有一些MVC设计模式的意思。这里介绍在其他类中要获取Doc对象的方法

在Cmianframe中可以使用

GetActiveDocument()

在Cview类中可以使用

这里要注意一点,在Cview类中包含了一个Cdocument的对象m_pDocument这个对象即使指向Doc类的基类对象的,而要实现基类对象到现在doc类对象的转换只要添加如下函数即可

inline CStockAppDoc* CStockAppView::GetDocument()

{ return (CStockAppDoc*)m_pDocument; }

则可以实现。

附上三个类对象相互访问方法:

1、主框架(CFrameWnd)中访问视图(CView)
CView* GetActiveView() const;
通常定义的视图为CView的派生类,在调用自定义视图对象的方法时

应该这样写:((CMouseKeyView*)GetActiveView())->MyFunc();

2、主框架(CFrameWnd)中访问文档(CDocument)
GetActiveDocument,返回CDocument对象;

3、在视图(CView)中访问文档(CDocument)
inline CMouseKeyDoc* CMouseKeyView::GetDocument()
{return (CMouseKeyDoc*)m_pDocument;}

4、在视图(CView)中访问框架(CFrameWnd)
CFrameWnd* GetParentFrame() const;

这里修改一下,因为上述代码获取的只是CMainFrame的父类对象CFrameWnd对象,要获取实际的CMainFrame对象可以进行如下操作首先在头文件中添加#include "MainFrm.h" 然后代码如下:

CMainFrame* frm=(CMainFrame*)::AfxGetMainWnd();frm->test(); 就能成功咯

5、在文档(CDocument)中访问框架(CFrameWnd)
CWnd* AfxGetMainWnd();
CWnd* AfxGetApp()->m_pMainWnd;

6、在文档(CDocument)中访问视图(CView)
UpdateAllViews
功能:通知所有的视图文档已被修改的信息
原型:
void UpdateAllViews(
CView* pSender, // 要更新的视图指针,如果希望更新所有视

图,将此参数设为NULL
LPARAM lHint=0L, // 包含更改消息的通知
CObject* pHint=NULL // 保管更改消息的对象
    }

7、在其他类中访问文档类(CDocument)
CDocument* GetDocument()
{
CFrameWnd* frm=(CFrameWnd*)::AfxGetMainWnd();
ASSERT(frm);
CDocument* pDoc=frm->GetActiveDocument();
ASSERT(pDoc);
ASSERT(pDoc->IsKindOf(RUNTIME_CLASS(CMouseKeyDoc)));
return (CMouseKeyDoc*)pDoc;
}

VC-基础:MFC单文档程序架构解析的更多相关文章

  1. MFC单文档程序架构解析

    MFC单文档程序架构解析 MFC单文档程序架构解析 这里我以科院杨老师的单文档程序来分析一下MFC单文档的程序架构,纯属个人见解,不当之处烦请指教! 首先我们了解到的是 图(一) theApp 是唯一 ...

  2. MFC单文档程序结构

    MFC单文档程序结构三方面: Doc MainFrame View

  3. 多线程串口编程工具CserialPort类(附VC基于MFC单文档协议通讯源程序及详细编程步骤)

    老有人觉得MSComm通讯控件很土,更有人大声疾呼:忘了它吧.确实当我们对串口编程有了一定的了解后,应该用API函数写一个属于自己的串口程序,由于编程者对程序了解,对程序修改自如.但我一直没有停止过用 ...

  4. MFC单文档程序添加HTML帮助支持

    1.在App类 构造函数中添加 EnableHtmlHelp(); 2.在Frame类中,添加消息影射: ON_COMMAND(ID_HELP_FINDER, CFrameWnd::OnHelpFin ...

  5. MFC学习(七) 单文档程序

    1 MFC单文档程序的主要类 (1)文档类(Document) 即应用程序处理的数据对象,文档一般从 MFC 中 CDocument 中派生.CDocument 类用于相应数据文件的读取以及存储 Cv ...

  6. MFC单文档

    一.创建并运行MFC单文档程序 1.创建单文档程序 这里使用的是VS2017.首先,打开VS2017,选择文件->新建->项目,然后选择Visual C++ -> MFC /ATL& ...

  7. VC++ MFC单文档应用程序SDI下调用glGenBuffersARB(1, &pbo)方法编译通过但执行时出错原因分析及解决办法:glewInit()初始化的错误

    1.问题症状 在VC++环境下,利用MFC单文档应用程序SDI下开发OpenGL程序,当调用glGenBuffersARB(1, &pbo)方法编译通过但执行时出错,出错代码如下: OpenG ...

  8. OpenCASCADE(一) VS2017+OpenCASCADE+MFC 下载配置安装运行单文档程序画个基本图形

    原文作者:aircraft 原文链接:https://www.cnblogs.com/DOMLX/p/12368154.html 一.下载OpenCASCADE 官网下载是: http://www.o ...

  9. MFC单文档框架分析及执行流程(转)

    原文转自 https://blog.csdn.net/u011619422/article/details/40402705 首先来分析一下MFC单文档类的结构: 它包括如下几个类: CAboutDl ...

随机推荐

  1. E20180506-hm

    更新: 2019/02/19 原来忘记分类,把此博文归入单词类 criterion n. 规范; (批评.判断等的) 标准,准则; criteria  n. (批评.判断等的) 标准,准则( crit ...

  2. HDOJ2579,BFS(三维数组标记)

    三维数组对于我这个萌萌的新手来说真是酷酷的,帅到不行,,, 三维数组前面还看到空间的哪一题HDOJ1240,这一题时间上的标记,更酷了!!! 不说废话了,贴一发代码: #include <std ...

  3. UGUI(四)事件系统的封装

    UGUI的事件系统默认使用时,是需要当前的类继承事件接口,就可以在该类中加入它的事件方法,但是在多人开发时这种方式没什么好处. 用过NGUI的或许都知道UIEventListener这个类,因为使用超 ...

  4. jq weui 滚动加载的坑

    1.一般情况下使用官网给个demo就可以了,如下: var loading = false; //状态标记 $(document.body).infinite().on("infinite& ...

  5. 新装ubuntu 12.04 , 使用技巧

    *********************************************** 一.让Ubuntu 12.04开机默认进入命令行模式. 修改 /etc/default/grubGRUB ...

  6. bzoj4300 绝世好题 【dp】By cellur925

    题目描述: 给定一个长度为\(n\)的数列\(a\),求\(a\)的子序列\(b\)的最长长度,满足bi&bi-1!=0(\(2<=i<=len\)). 90分做法: 并没有部分分 ...

  7. html 5 video audio

    autoplay autoplay 如果出现该属性,则视频在就绪后马上播放. controls controls 如果出现该属性,则向用户显示控件,比如播放按钮. height pixels 设置视频 ...

  8. [題解](縮點)luogu_P2341受歡迎的牛

    對於每個強聯通分量,這些牛一定都互相喜歡,所以縮點(我也不知道怎麼想到的) 接下來就是統計答案,最後縮成了一個DAG圖,如果這個點是明星的話,其他每個點一定直接或間接的鏈接這個點 也就是說其他點一定有 ...

  9. Codeforces 1139F(树状数组+扫描线)

    题目传送 做法 对于每个人,inc为x,pref为y:对于每道菜,p和s为x,b为y 于是根据题意有\[p[i]<=x<=s[i]\]\[p[i]+b[i]<=x+y\]\[p[i] ...

  10. git stash暂存当前正在进行的工作

    git stash 可用来暂存当前正在进行的工作, 比如想pull 最新代码, 又不想加新commit, 或者另外一种情况,为了fix 一个紧急的bug,  先stash, 使返回到自己上一个comm ...