VC文件打开、保存与关闭菜单命令(转)

下午和黄老师 琢磨了一下 打开,发现没有理清楚,所以转载了一篇来学习学习。

 

http://hi.baidu.com/%CA%C5%C8%A5%B5%C4%B7%AB/blog/item/2b70a1438169741b9313c68b.html
VC文件打开、保存与关闭菜单命令(转)

第一部分:


五个命令ID: 处理函数  
ID_FILE_NEW CWinApp::OnFileNew   
ID_FILE_OPEN    CWinApp::OnFileOpen  
ID_FILE_SAVE    CDocument::OnFileSave  
ID_FILE_SAVEAS  CDocument::OnFileSaveAs  
ID_FILE_CLOSE   CDocument::OnFileClose  

1.ID_FILE_NEW  
CWinApp::OnFileNew调用CDocManager::OnFileNew。  
|  
CDocManager::OnFileNew判断文档模板是否多于一个,是则显示文档类型对话框(AFX_IDD_NEWTYPEDLG)  
让用户选择要创建的文档类型。然后调用CDocTemplate::OpenDocumentFile(NULL)。  
|  
CDocTemplate::OpenDocumentFile(LPCTSTR lpszPathName,BOOL bMake_Visible=TRUE)对于SDI和MDI的  
处理不一样。  

对于SDI,1,若已有文档打开,对其重新初始化,调用CDocument::SaveModified()保存当前文档;若没  
有文档存在,则调用CreateNewDocument()创建文档对象,再调用CreateNewFrame(pDoucment,NULL)创建  
文档的框架窗口。2,若lpszPathName为NULL,则调用CDocument::OnNewDocument()处理新文档,用  
pDocument->SetPathName(lpszPathName)设置文档的路径。3,判断当前线程主框架窗口是否存在,不存  
在则将1中创建的新框架作为主框架。4,调用InitialUpdateFrame显示框架窗口。  

对于MDI,与SDI基本相同,但是没有第3步。  
|  
CDocument::OnNewDocument()首先调用DeleteContents()删除原文档内容,使用m_strPathName.Empty()  
清除当前文档路径,SetModifiedFlag(FALSE)。  

2.ID_FILE_OPEN  
CWinApp::OnFileOpen调用CDocManager::OnFileOpen。  
|  
CDocManager::OnFileOpen首先显示文件打开对话框(AFX_IDS_OPENFILE),然后调用CWinApp::  
OpenDocumentFile(FileName)。  
|  
CWinApp::OpenDocumentFile(FileName)调用CDocManager::OpenDocumentFile。  
|  
CDocManager::OpenDocumentFile(LPCTSTR lpszFileName)遍历文档模板,对每个模板用  
MatchDocType(szPath,pOpenDocument)匹配文档类型。匹配时主要根据文件扩展名判断。若文件已经在  
某个文档中打开,则激活文档的第一个视图,否则用匹配的文档模板pBestTemplate->OpenDocumentFile  
(szPath)。  
|  
CDocTemplate::OpenDocumentFile调用CDocument::OnOpenDocument打开文件。  
|  
CDocument::OnOpenDocument打开文件,用DeleteContents删除现有文档内容,创建文件对应的CArchive  
对象loadArchive,Serialize(loadArchive)读文件内容,SetModifiedFlage(FALSE)。  
MDI: New Frame, New View?  
SDI: Update View?  

3.ID_FILE_SAVE  
CDocument::OnFileSave调用CDocument::DoFileSave()。  
|  
CDocument::OnFileSave判断文件是否只读,是则DoSave(NULL),否则DoSave(m_strPathName)。  
|  
CDocument::DoSave(LPCTSTR lpszPathName,BOOL bReplace=TRUE)的流程为:1,判断lpszPathName不空  
则跳第二步,为空则获取文档的路径名,加上扩展符。bReplace为TRUE则显示保存文件对话框(AFX_IDS_  
SAVEFILE),否则显示拷贝保存对话框(AFX_IDS+SAVEFILECOPY)。2,调用CDocument::OnSaveDocument保  
存文档。3,若bReplace==TRUE则SetPathName(newName)覆盖当前路径名。  
|  
CDocument::OnSaveDocument打开文件,创建CArchive对象saveArchive,Serialize(saveArchive)读写  
文件,SetModifiedFlag(FALSE)。  

4.ID_FILE_SAVEAS  
CDocument::OnFileSave调用DoSave(NULL)。  

5.ID_FILE_CLOSE  
CDocument::OnFileClose调用SaveModified()保存文件,再用OnCloseDocument处理文档关闭。  
|  
CDocument::SaveModified()用IsModified()判断是否修改,是则显示文件保存对话框(AFX_IDP_ASK_TO_  
SAVE),若用户选择“是”,则调用DoFileSave()保存文件。  
|  
CDocument::OnCloseDocument()首先销毁文档框架,再用DeleteContents()删除文档内容,若m_  
bAutoDelete则delete this。  

第二部分:


MDI打开文件的过程  
CWinApp::OnFileOpen()  
{  
   CDocManager::OnFileOpen()  
   {  
       CDocManager::DoPromptFileName()  
       {  
           CFileDialog::DoModal();  
       }  
       CWinApp::OpenDocumentFile()  
       {  
           ...选择文档模板;  
           ...调整视图和框架;  
           CDocTemplate::OpenDocumentFile();  
           {  
               判断有无现存文档,是否需要保存  
               新建框架  
               判断文件是否存在,调用CMyDoc::OnOpenDocunment/OnNewDocument  
           }  
       }  
   }  
}  
----------------------------------------  
Document/View命令处理流程  
ID_FILE_NEW  
CWinApp::OnFileNew()  
{  
   CDocManager::OnFileNew()  
   {  
       if(没有文档模板)  
           return;  
       if(有多个文档模板)  
       {  
           弹出对话框让用户选择;  
           取得模板指针;  
       }  
       CMultiDocTemplate::OpenDocumentFile()  
       {  
           new一个文档;  
           创建子框架;  
           构建frame,doc,view之间的关系;  
           CDocument::OnNewDocument()  
           {  
               DeleteContents();  
           }  
           InitialUpdateFrame();  
           return pDoc;  
       }  
   }  
}  

ID_FILE_OPEN  
CWinApp::OnFileOpen()  
{  
   CDocManager::OnFileOpen()  
   {  
       CDocManager::DoPromptFileName()  
       {  
           CFileDialog::DoModal();  
       }  
       CWinApp::OpenDocumentFile()  
       {  
           ...选择文档模板;  
           ...调整视图和框架;  
           CDocTemplate::OpenDocumentFile();  
           {  
               判断有无现存文档,是否需要保存  
               新建框架  
               判断文件是否存在  
               CMyDoc::OnOpenDocunment()  
               {  
                   打开文件;  
                   DeleteContents();  
                   建立CArchive;  
                   Serialize;  
                   关闭CArchive;  
               }  
           }  
       }  
   }  
}  

ID_FILE_SAVE  
CDocument::OnFileSave()  
{  
   DoFileSave(NULL)  
   {  
       CFileDialog::DoModal();  
       OnSaveDocument()  
       {  
           CArchive;  
           Serialize;  
       }  
   }  

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

在这里将讲述SDI程序中application object、the main frame window、the document、the view、the document template object以及the associate string and menu resources之间的关系。

The Windows Application Object
在CWinApp派生类的Implement文件中会有类似CMyApp theApp的语句。theApp是一个全局变量,它就是启动MFC应用程序的机制所在。
以下是MFC应用程序启动的摘要:
1、Windows把应用程序加载到内存;
2、构造全局变量theApp(所有全局变量在程序被载入内存时被构造);
3、Windows调用全局函数WinMain,它是MFC的一部分,等同于无窗口应用程序的main函数---主程序的入口;
4、WinMain寻找到CWinApp派生类的对象theApp;
5、WinMain为theApp调用InitInstance函数,InitInstance函数可以被重载;
6、重载的InitInstance函数启动加载document、显示the main frame and view windows;
7、WinMain为theApp启动Run函数,Run函数负责分派window messages和command messages。

The Document Template Class
在CWinApp派生类的InitInstance中会有如下代码:
CSingleDocTemplate* pDocTemplate;
pDocTemplate = new CSingleDocTemplate(
   IDR_MAINFRAME,
   RUNTIME_CLASS(CStudentDoc),
   RUNTIME_CLASS(CMainFrame),       // main SDI frame window
   RUNTIME_CLASS(CStudentView));
AddDocTemplate(pDocTemplate);
这段代码建立了the application class、the document class、the view window class、the main frame window四个类之间的联系。这时,application class的对象已经存在(theApp),但其他四个类的对象还没有构造,MFC的动态创建机制就起到了作用。
一下两个图分别表示了以上四个类之间的关系、四个类的对象之间的关系:



Creating an Empty Document—The CWinApp::OnFileNew Function
当application class的InitInstance调用完AddDocTemplate以后,它将调用OnFileNew。
OnFileNew做如下事情:
1、构造Document对象,但不从硬盘中读取数据;
2、构造the main frame对象和the main frame window,但不显示它,它包括IDR_MAINFRAME、the toolbar、the status bar;
3、构造view对象和view窗口,但不显示它;
4、建立the document、main frame、view对象之间的关系,这里要借助AddDocTemplate建立的类关系;
5、为document对象调用虚函数CDocument::OnNewDocument,CDocument::OnNewDocument函数将调用虚函数DeleteContent函数;
6、为view对象调用虚函数CView::OnInitialUpdate;
7、为frame对象调用CFrameWnd::ActivateFrame以显示the main frame window(有the menus,view window,control bars)。

The Document Class's OnNewDocument Function
构造函数所完成的工作越少越好,这样构造函数出错的几率就会越来越小。而CDocument::OnNewDocument 和 CView::OnInitialUpdate 是做初始化的好地方。在这里你可以弹出一个MessageBox。如果出错,OnNewDocument还可以返回FALSE。值得注意的是,OnNewDocument可以被调用很多次,所以有些指令需只执行一次的话,就需要做一个“first time” flag成员变量。

Connecting File Open to Your Serialization Code—The OnFileOpen Function
文件打开菜单映射到CWinApp::OnFileOpen函数,它将执行以下几个步骤:
1、提示用户选择一个文件;
2、调用虚函数CDocument::OnOpenDocument,CDocument::OnOpenDocument打开文件,调用 CDocument::DeleteContents,并且构造一个CArchive对象。随后调用document的Serialize函数从 archive中载入数据;
3、调用view的OnInitialUpdate函数。

Connecting File Save and File Save As to Your Serialization Code
文件保存和另存为菜单映射到CDocument的OnFileSave函数,OnFileSave函数依次调用Serialize函数,它使用一个archive对象进行存储。

The Document's "Dirty" Flag
CDocument有一个protected成员函数m_bModified。可以通过CDocument的函数SetModifiedFlag和 IsModified访问m_bModified。当新建一个document或者读取完硬盘上的数据后,m_bModified被 Application Framework设为FALSE。当document的内容被改写后要确保把m_bModified设为TRUE。

小结:OnFileNew和OnFileOpen都在Application类中,Application类里面有DocumentTemplate,可以同时调动DocumentView(OnInitialUpdate)。
OnFileSave和OnFileSaveAs都在Document类中,它只需做Serialization,不牵扯View的操作,故不需要把它们放在Application类中。

你可能感兴趣的:(框架,File,serialization,application,Class,文档)