MFC文档、视图、框架的建立

转自:http://ffwmxr.blog.163.com/blog/static/66372722201001604630889/

文档将描述MFC框架程序中WinApp , Document Templates, Document, View , Frame创建和所有关系。

CWinApp
在一个系统程序中只有一个CWinApp对象,该对象会被静态的创建并在MFC框架内部实现的WinMain()中进行初始化(动态链接库程序中不会包含一个CWinApp实例且程序的初始化是在DllMain()中进行)

CWinApp负责管理一个Document Template的链表(CPtrList), 在一个程序中通常会有一个或多个 Document Template,这些Document Template一般是在CWinApp类的虚函数InitInstance中从资源文件中加载的
同时WinApp对象拥有全部的框架窗体(Frame Window),主框架窗体(main frame window)必需被记录保存在WinApp对象的m_pMainWnd中;如果想实现个人的设定的化,可以在InitInstance的实现中改写预先由向导设定的m_pMainWnd;一般的,主框架窗体在SDI中为CMainFrame, 而在MDI中则是CMDIFrameWnd

CDoTemplate
Document Template负责管理Ducument的创建和管理, 它拥有全部全部由它创建的Document

在SDI中 CSingleDocTemplate负责跟踪唯一打开的Document
在MDI中CMultiDocTemplate用一个链表(CPtrList)来负责维护全部被打开的Document
CDocTemplate提供虚成员函数来增加或删除文档
CDocTemplate::AddDocument()
CDocTemplate::RemoveDocument()

同时CDocumentTemplate是CDocument的友元类,在CDocument中一个CDocumentTemplate的指针用来指向创建文档的Document Template

CWinApp负责实现OnFileOpen , 将会依次查询全部Document Template

CDocTemplate同时负责与Document和框架窗体关联的界面
CDocTemlate负责新建立的未命名文件的计数 将其命名为untitled X

CDocument
Document是为CDocument Template所有的
Document 负责维护当前显示文档的视图的链表(CPtrList)
Document不负责创建视图但是他们会在视图创建后关联在视图上,且当一个文档关闭(file open /close)时所有的关联的视图都将被关闭;同时最后一个与文档关联的视图关闭时文档也会被关闭
Document 提供结构AddView()   RemoveView()来维护视图链表
同时CDocument是CView的友元
在CView中的成员变量CView::m_pDocument用来指向视图所关联的文档

CFrameWnd
包括一些列的派生类CMDIFrameWnd CMDIChildWnd
负责创建在框架的客户区的窗体

***************************************************************************************************************************

遍历应用程序的视图,文档,文档模板

转载自:http://blog.csdn.net/SammyLan/archive/2006/04/07/654341.aspx

void EnumAllView(CDocument*pDoc)

// 遍历文档关联的视图

{

    POSITION pos=pDoc->GetFirstViewPosition();

    while (pos)

    {

       CView*pView=pDoc->GetNextView(pos);

       /**/

    }

}

void EnumAllDocment(CDocTemplate*pDocTemp)

// 遍历文档模板对应的文档

{

    POSITION pos=pDocTemp->GetFirstDocPosition();

    while (pos)

    {

       CDocument*pDoc=pDocTemp->GetNextDoc(pos);

       if (pDoc)

       {

           EnumAllView(pDoc);

       }

    }

}

void EnumAllDocTemp()

// 遍历应用程序的文档模板

{

    CDocManager*pDocMan=AfxGetApp()->m_pDocManager;

    if (!pDocMan)

    {

       return ;

    }

    POSITION pos=pDocMan->GetFirstDocTemplatePosition();

    while (pos)

    {

       CDocTemplate*pDocTemp=pDocMan->GetNextDocTemplate(pos);

       if (pDocTemp)

       {

           EnumAllDocment(pDocTemp);

       }

    }

}

***************************************************视图中的一些操作*****************************************************

    //更新所有视图:

    pDoc->UpdateAllViews();

    //添加和移除视图

    pDoc->AddView(pView);

    pDoc->RemoveView(pView);

    //无论何时文档数据发生更改,都应该调用

    pDoc->SetModifiedFlag();

    //同时可以调用

    pDoc->IsModified();//获取状态

    //应用程序可以通过SetTitle成员函数设置文档对象的标题,也既可以使用SetPathName函数为文档设置完全限定名,用GetPathName函数获取全路径名.可以通过GetDocTemplate获取程序在创建文档的时候将其关联到的文档模板

    在CDocTemplate中有一个GetDocString成员函数,可以查询IDR_MAINFRAME字符串的值,注意,没有对应的SetDocString函数。

***************************************************************************************************************************

框架、文档与视图的毁灭

转载自:http://blog.csdn.net/fiolee/archive/2004/08/11/71714.aspx

1、  无文档的多视图程序的毁灭顺序及解决方法

在无文档的多视图程序中,程序主框架、子框架、及视图,在通常情况下,如果你直接关闭程序主框架,程序直接调用destroywindow,子框架及视图则自动清理,但程序不会调用子框架的destroywindow以及postncdestroy两个函数,如果你先关闭视图,关闭的顺序如下,程序先调用子框架的的destroywindow,然后调用视图的postncdestroy,接着就是子框架的postncdestroy,程序里没有调用到视图的destroywindow。

有些情况下,必须保证程序的先子框架关闭,这时候上面这个顺序就会造成程序出错了,正如前面所说,直接关闭主框架时只调用了一个destroywindow函数,其余的函数都不执行了。解决的思路,我们得把程序的毁灭顺序改一改,把子框架的毁灭放在主框架毁灭之前:

CChildFrame *pChildFrame = (CChildFrame *)this->GetActiveFrame();

while (pChildFrame)

{

        pChildFrame->DestroyWindow();

        pChildFrame = (CChildFrame *)this->GetActiveFrame();

}

通过此循环,可以将所有子框架删除,在循环中,调用的GetActiveFrame函数是获得当前活动子框架,在删除当前活动子框架时,其它的子框架又会变成活动子框架,所以这种方法不会造成死循环。

2、  多文档视图程序的毁灭顺序

它和上面的情况差不多,只是多了一个文档,它的毁灭是在子框架的OnClose之后,子框架的destroywindow之前。

*******************************************************************************************************************************************

文档、文档模板、视图和框架类的互相访问

从该对象 如何访问其他对象
全局函数 调用全局函数AfxGetApp可以得到CWinApp应用类指针
应用 AfxGetApp()->m_pMainWnd为框架窗口指针;用CWinApp::GetFirstDocTemplatePostion、CWinApp::GetNextDocTemplate来遍历所有文档模板
文档 调用CDocument::GetFirstViewPosition,CDocument::GetNextView来遍历所有和文档关联的视图;调用CDocument:: GetDocTemplate 获取文档模板指针
文档模板 调用CDocTemplate::GetFirstDocPosition、CDocTemplate::GetNextDoc来遍历所有对应文档
视图 调用CView::GetDocument 得到对应的文档指针;调用CView::GetParentFrame 获取框架窗口
文档框架窗口 调用CFrameWnd::GetActiveView 获取当前得到当前活动视图指针;调用CFrameWnd::GetActiveDocument 获取附加到当前视图的文档指针
MDI 框架窗口 调用CMDIFrameWnd::MDIGetActive 获取当前活动的MDI子窗口(CMDIChildWnd)


  我们列举一个例子,综合应用上表中的函数,写一段代码,它完成遍历文档模板、文档和视图的功能:

CMyApp *pMyApp = (CMyApp*)AfxGetApp(); // 得到应用程序指针
POSITION p = pMyApp->GetFirstDocTemplatePosition();// 得到第1个文档模板
while (p != NULL) // 遍历文档模板
{
 CDocTemplate *pDocTemplate = pMyApp->GetNextDocTemplate(p);
 POSITION p1 = pDocTemplate->GetFirstDocPosition();// 得到文档模板对应的第1个文档
 while (p1 != NULL) // 遍历文档模板对应的文档
 {
  CDocument *pDocument = pDocTemplate->GetNextDoc(p1);
  POSITION p2 = pDocument->GetFirstViewPosition(); // 得到文档对应的第1个视图
  while (p2 != NULL) // 遍历文档对应的视图
  {
   CView *pView = pDocument->GetNextView(p2);
  }
 }
}


  由此可见,下面的管理关系和实现途径都是完全类似的:

  (1)应用程序之于文档模板;

  (2)文档模板之于文档;

  (3)文档之于视图。

  下面的图1、图2 分别给出了一个多文档/视图框架MFC程序的组成以及其中所包含类的层次关系。

MFC文档、视图、框架的建立_第1张图片
图1 多文档/视图框架MFC程序的组成


MFC文档、视图、框架的建立_第2张图片
图2 文档/视图框架程序类的层次关系

你可能感兴趣的:(MFC文档、视图、框架的建立)