Visual C++应用框架揭密
北京理工大学光电工程系 卢英威
电脑编程技巧与维护 1999.1 P30
Visual C++应用框架提供的强大功能,为我们的程序开发提供了极大的方便,利用其应用生成器可以很轻松地生成应用程序的框架.许多常用功能,例如文档的创建,文件的打开,保存等操作的大部分代码都由应用框架来完成.这些操作对于开发软件的用户而言,是透明的. Microsoft公司的设计Visual C++的初衷是尽量向用户提供尽可能简单的接口,但是,正是因为应用框架的这一透明性,使Visual C++的运行流程显得非常神秘,当涉及这些Visual C++已经提供的功能模块的软件开发时,用户(特别是那些刚刚接触Visual C++的)不知道什么时候该使用什么接口函数,想改变Visual C++的一些默认操作时也不知道该从哪儿下手,使开发的难度增加.
本人在编程实践中发现,可以通过Visual C++自带的强大的Debug调试工具,跟踪应用框架代码,从而得出应用框架操作的部分流程.具体实现是:利用Visual C++应用生成器生成应用框架代码,启动 Classwizard,在各个类中加入欲跟踪的接口函数,编辑接口函数,设置断点,再启动Debug进行单步跟踪调试.利用这一方法,我们还可以跟踪 Visual C++提供的例子代码.在跟踪过程中,Visual C++的核心代码是禁止跟踪调试的,我们只要跳过就行.
以下是本人对多文档应用程序的文档创建,文件的打开, 保存(另存为)操作的跟踪结果,单文档应用与多文档应用的操作流程基本相似,只是用CSingleDocTemplate类的同名函数来替代对 CMultiDocTemplate类同名函数的调用.以下说明中,"应用程序"均指用户生成的基于应用框架的实际程序,"用户代码"指用户在接口函数中加入的代码.
一.创建文档
ON_COMMAND(ID_FILE_NEW,CWinApp::OnFileNew)
入口:CWinApp::OnFileNew
调用:CdocManager::OnFileNew()
1.判断有无文档模板,无则函数返回;
2.判断是否有多个文档模板,如有则启动文档模板选择对话框, 让用户选择一个模板后,返回模板指针.
3.由模板指针访问CMultiDocTemplate::OpenDocumentFile()
31.创建新文档对象
32.创建与新文档对象对应的子框架,构筑子框架,文档和视窗之间的关系
33.调用应用程序的OnNewDocument()
a.调用CDocument::OnNewDocument()
调用应用程序的DeleteContents();
b.用户代码
34.调用InitialUpdateFrame()显示视窗
35.返回文档指针
4.返回
二.打开文档
ON_COMMAND(ID_FILE_OPEN,CWinApp::OnFileOpen)
入口:CWinApp::OnFileOpen
调用:CDocManager::OnFileOpen()
1.弹出对话框让用户选择待打开的文件,返回该文件的全路径名称, 供下面函数调用时使用.
2.调用应用程序的OpenDocumentFile(LPCTSTR lpszFileName)
21.用户代码
22.调用CWinApp::OpenDocumentFile(lpszFileName),
调用CWinApp::OpenDocumentFile(lpszFileName),
调用CDocManager::OpenDocumentFile
a.判断有无该文件的打开文档对象,有则激活该窗口并返回
b.调用CMultiDocTemplate::OpenDocumentFile()
b1.创建新的文档对象,创建新文档对象对应的子框架
b2.调用应用程序的OnOpenDocument()
(1)调用CDocument::OnOpenDocument()
.打开文件对象
.调用应用程序的DeleteDontents();
.建立与此文件对象相关联的CArchive对象
.调用应用程序文档对象的Serialize()函数
.关闭CArchive对象,文件对象
(2)显示该文件窗口
b3.返回
三.文件保存
ON_COMMAND(ID_FILE_SAVE,CDocument::OnFileSave)
入口:CDocument::OnFileSave()
调用:CDocument::DoFileSave()
如果当前文档对应的文件名为空或者为只读文件,以NULL 为参数调用CDocument::DoSave(NULL),否则,以当前文档对应的文件名为参数调用CDocument::DoSave(m_strPathName);
ON_COMMAND(ID_FILE_SAVEAS,CDocument::OnFileSaveAs)
入口:CDocument::OnFileSaveAs()
以NULL为参数直接调用CDocument::DoSave(NULL);
CDocument::DoSave(LPCTSTR lpszPathName,BOOL bReplace);
如果lpszPathName为NULL,
1.设定应用程序默认的文件名
2.打开通用文件保存对话框,由用户设定保存文件的名字,并返回该文件名
如果lpszPahtName不为NULL,则跳为以上两步,直接进入下一步
3.调用应用程序的OnSaveDocument()
31.用户代码
32.调用CDocument::OnSaveDocument()
a.创建或打开文件对象
b.建立相对应的CArchive对象
c.调用应用程序文档对象的序列化函数Serialize()
d.关闭文件对象,CArchive对象
e.设置文件未修改标志
4.返回