友情提示:如果本网页打开太慢或显示不完整,请尝试鼠标右键“刷新”本网页!
第三电子书 返回本书目录 加入书签 我的书架 我的书签 TXT全本下载 『收藏到我的浏览器』

VC语言6.0程序设计从入门到精通-第8部分

快捷操作: 按键盘上方向键 ← 或 → 可快速上下翻页 按键盘上的 Enter 键可回到本书目录页 按键盘上方向键 ↑ 可回到本页顶部! 如果本书没有阅读完,想下次继续接着阅读,可使用上方 "收藏到我的浏览器" 功能 和 "加入书签" 功能!




         



              // Register the application’s document templates。    Document templates  



              //    serve as the connection between documents; frame windows and views。  



         



              CSingleDocTemplate* pDocTemplate;  



              pDocTemplate = new CSingleDocTemplate(  



                      IDR_MAINFRAME;  



                      RUNTIME_CLASS(CHelloMFCDoc);  



                      RUNTIME_CLASS(CMainFrame);              // main SDI frame window  



                      RUNTIME_CLASS(CHelloMFCView));  



              AddDocTemplate(pDocTemplate);  



              // Parse mand line for standard shell mands; DDE; file open  



              CmandLineInfo cmdInfo;  



              ParsemandLine(cmdInfo);  



              // Dispatch mands specified on the mand line  



              if (!ProcessShellmand(cmdInfo))  



                      return FALSE;  



         



              // The one and only window has been initialized; so show and update it。  



              m_pMainWnd…》ShowWindow(SW_SHOW);  



              m_pMainWnd…》UpdateWindow();  



              return TRUE;  



       }  



       从上面的代码可以看出,AppWizard  自动生成的InitInstance()函数主要完成下述功能。  



 ·20 ·  


…………………………………………………………Page 30……………………………………………………………

                                                  第 2 章    应用程序基本框架  



   o  从。ini 文件或 Windows 注册表中获取一些标准的文件选项,包括取得最近使用的文件 

     名称,以便在应用程序的文件菜单中列出,供用户快速打开最近编辑的文件。  

   o  该函数的中心工作是构造文档模板类(单文档或多文档模板类)的对象,指明了该文 

     档模板对象对应的文档类、框架窗口类和视图类。这使得以后可以据此构造文档类、 

     边框窗口类和视图类的对象及创建新窗口。  

   o  如果是 MDI 应用程序,则构造主边框窗口类的对象并创建主边框窗口。  

   o  另一重要工作是调用 ProcessShellmand() 函数进行程序窗口启动方式的分析处理, 

     如果在其中程序的命令行中提供了文档文件名,则启动程序时,打开该文件;如果没 

     有提供文档文件名,则新建一个空文件。  

   此外,InitInstance()函数还可执行其他一些功能,这里就不一一说明了。Windows 允许在 

同一时刻运行程序的几份拷贝。在概念上,应用程序的初始化可以被分为两个部分:一次性 

的应用程序初始化工作和示例的初始化工作,前者在应用程序第一次运行时完成,后者在每 

次运行程序时都会执行。框架中  WinMain() 函数重载  InitInstance()函数以初始化在 Windows 

下运行的应用程序的每个新实例。通常,重载                  InitInstance 以构造主窗口对象并设置 

CWinThread::m_pMainWnd 数据成员,使其指向这个窗口。  



   技巧:快速显示当前函数相关的信息  



        在编写代码或查看代码的时候,常常需要查看某个  MFC  类库中定义的函数的相关信息。 



        可以通过快捷键 F1 来实现,具体做法是在 IDE 客户区将鼠标光标放在要查看函数名称处 



        并停止鼠标光标移动,按 F1 键,这时将启动 MSDN  Libray 并切换到要查看的函数信息页 



        面上。  



2。3。2    Run()函数  



   应用程序的大部分时间都是在应用程序类的 Run()成员函数中进行处理。WinMain() 函数 

在初始化应用程序实例后,就调用 Run() 函数来处理消息循环。  



                                               



                       图 2…16    Run 成员函数的消息循环  



                                                                ·21 ·  


…………………………………………………………Page 31……………………………………………………………

Visual C++ 6。0 程序设计从入门到精通  



    Run()成员函数不断执行消息循环,检查消息队列中有没有消息。如果有消息,Run() 函 

数将其派遣,交由框架去处理,然后返回继续消息循环。如果没有消息,Run() 函数将调用 

OnIdle()函数来执行用户或框架在空闲时要执行的工作,如用户接口更新消息处理等。如果既 

没有消息要处理,也没有空闲时的处理工作要做,则应用程序将一直等待,直到有事件发生。 

应用程序结束时, 

                 Run() 函数将调用 ExitInstance()函数使应用程序退出。消息循环的流程图如 

图 2…16 所示。  

    Run() 函数很少被重载,但是也可以重载它以提供特殊的功能,HelloMFC 中的默认 Run() 

函数定义如下:  



    int CHelloMFCApp::Run()    



     {  



            return CWinApp::Run();  



     }  



2。3。3    ExitInstance() 函数  



    ExitInstance()函数是在用户退出应用程序的运行实例时由 Run() 函数调用的。  

    框架在  Run()成员函数内部调用这个函数以退出应用程序的实例。此函数只能在  Run() 

成员函数内部调用。这个函数的默认实现将框架的选项写入应用程序的。ini  文件。重载这个 

函数可以在应用程序退出的时候执行一些清除操作。  

    HelloMFC 中默认的 ExitInstance()函数定义如下:  



    int CHelloMFCApp::ExitInstance()    



     {  



         return CWinApp::ExitInstance();  



     }  



    返回值表示应用程序的退出码,0 表示没有错误,大于 0  的值表示有错误。这个值被用 

作 WinMain 的返回值。  



2。3。4    OnIdle() 函数  



    OnIdle()函数是在应用程序的消息队列中没有消息时由 Run() 函数调用的。  

    如果要执行空闲时处理,则必须重载这个成员函数。当应用程序的消息队列为空时, 

OnIdle 就在默认的消息循环中被调用,可以重载这个函数来调用后台空闲处理任务。  

    HelloMFC 中默认的 OnIdle()函数定义如下:  



    BOOL CHelloMFCApp::OnIdle(LONG lCount)    



     {  



         return CWinApp::OnIdle(lCount);  



     }  



    其中  lCount  参数是一个计数值,当应用程序的消息队列为空,OnIdle()函数被调用时, 

该计数值就增加  1。每当一条新消息被处理时,该计数值就被复位为 0,可以使用 lCount 参 

数来确定应用程序不处理消息时空闲时间的相对长度。  



 ·22 ·  


…………………………………………………………Page 32……………………………………………………………

                                                   第 2 章    应用程序基本框架  



   OnIdle()函数应返回 0  以表明不需要更多的空闲处理时间。当消息队列为空时,OnIdle() 

每被调用一次 lCount 参数就增加 1,而每处理一条新消息 lCount 就被复位为 0,可以根据这 

个计数值调用不同的空闲处理例程。  

    空闲循环处理的过程如下。  

   o  如果  MFC  类库中的消息循环检查消息队列并发现没有未被处理的消息,就为应用程 

     序对象调用 OnIdle()函数,并将 lCount 参数设为 0 。  

   o  OnIdle()函数执行一些处理,然后返回一个非零值,表示它还需要被调用,以进行进一 

     步处理。  

   o  消息循环再次检查消息队列,如果没有未处理的消息,则再次调用  OnIdle()函数,增 

     加 lCount 参数。  

   o  OnIdle()函数结束所有的空闲任务并返回 0,这告诉消息循环停止调用 OnIdle()函数直 

     到在消息队列中接收到下一条消息为止,在那时,空闲循环将重新启动,而参数被设 

     为 0 。  

    由于只有在 OnIdle 返回之后应用程序才能处理用户输入,因此在 OnIdle  中不应执行较 

长的任务。  



   注意:OnIdle 除可实现更新用户接口对象(如菜单项和工具条等)外,还实现了内部数据结构的 



        清理。因此,如果重载了       OnIdle 函数,必须用重载版本中使用的       lCount 值来调用 



        CWinApp::OnIdle 。首先调用所有基类的空闲处理(即直到基类的 OnIdle 返回 0 ),如果需 



        要在基类处理完成之前进行一些工作,则应重复基类的实现以在工作期间选择一个合适的 



        lCount 值。  



2。4    文档类和视图类  



   MFC  AppWizard 自动生成的应用程序默认采用文档…视图结构,因为一般应用程序都要 

对某种文档进行处理(文档并不一定是文件,但通常可以理解为文件),而文档又通过视图与 

用户打交道(或称交互),文档的内容通过视图窗口显示给用户,用户在视图中对文档所作的 

修改由视图通知文档对象,视图实际上充当了一个中介者的角色。  

   下面将对文档视图结构中文档类和视图类及它们的相互关系作相应介绍。  



2。4。1    文档类  



   文档类(CDocument )在 MFC 类库中的层次结构如图 2…17 所示。  



                                            



                     图 2…17    CDocument 在 MFC 类库中的位置  



    不管是  SDI  应用程序还是  MDI  应用程序,文档类都是从  CDocument  类派生出来的,      



                                                                 ·23 ·  


…………………………………………………………Page 33……………………………………………………………

Visual C++ 6。0 程序设计从入门到精通  



App  Wizard  自动生成的文档类为 HelloMFCDoc ,其定义在文件 HelloMFCDoc。h  中,实现在 

HelloMFCDoc。cpp 中。  

     若要使用 AppWizard 提供的文档类,必须执行下列操作。  

     o  为每个文档类型从 Cdocument 中派生一类。  

     o  添加成员变量以存储每个文档的数据。  

     o  在文档类中重写 CDocument 的 Serialize 成员函数,Serialize 用于从磁盘读取文档的数 

        据和将文档数据写入磁盘。  

     o  可能还需要重写其他                CDocument    成员函数,如经常需要重写                     OnNewDocument       和 

        OnOpenDocument 以初始化文档的数据成员、重写 DeleteContents  以动态销毁分配的数 

        据。  

     在 HelloMFCDoc 类中,主要重载了基类的两个函数:OnNewDocument()和 Serialize()。 

其中 OnNewDocument()函数用于文档对象的初始化,Seralize()函数用于实现序列化。虽然定 

义了这两个函数,但只是提供了一个框架,具体的函数内容需要用户自己加入。  

     处理文档序列化的 OnSerialize()函数的默认代码如下:  



     void CHelloMFCDoc::Serialize(CArchive& ar)  



     {  



          //存储  



          if (ar。IsStoring())  



          {  



               // TODO: add storing code here  



          }  



          //读取  



          else  



          {  



               // TODO: add loading code here  



          }  



     }  



     技巧:当文档被修改时,在其标题上加上“*”作标志。  



     文档类还可处理由菜单项、工具栏按钮或快捷键生成的某些命令。默认情况下, 

CDocument  使用序列化方式处理“File|Save ”和“File|Save  as ”菜单命令,文档可以有消息 

映射,但与视图不同,文档无法处理标准 Windows 消息,而只能处理 WM_MAND 命令 

消息或命令。  



     实例 2…3 :技巧演示程序。源代码在光盘中“02实例 2…3EditApp ”目录下。  



     如重载 CEditAppDoc 类的 SetModifiedFlag 函数,代码如下:  



     void CEditAppDoc::SetModifiedFlag(BOOL bModified /* = TRUE */)  



     {  



             CString strTitle = GetTitle();  



          CString strDirtyFlag = 〃 *〃; // note space before the ’*’  



 ·24 ·  


…………………………………………………………Page 34……………………………………………………………

                                                                                                            第 2 章    应用程序基本框架  



                               // so we don’t break Save As dialog  



         



               if (!IsModified() && bModified)  



               {  



                       SetTitle(strTitle + strDirtyFlag);  



               }  



               else if ( IsModified() && !bModified )  



               {  



                       int nTitleLength = strTitle。GetLength();  



                       int nDirtyLength = strDirtyFlag。GetLength();  



                       SetTitle( strTitle。Left(nTitleLength nDirtyLength) );  



               }  



               UpdateFrameCounts();  



               CDocument::SetModifiedFlag(bModified);  



                 



       }  



        同时映射 CEditAppView 的 EN_CHANGE 消息如下:  



       void CEditAppView::OnChange()    



        {  



               // TODO: If this is a RICHEDIT control; the control will not  



               // send this notific
返回目录 上一页 下一页 回到顶部 1 2
快捷操作: 按键盘上方向键 ← 或 → 可快速上下翻页 按键盘上的 Enter 键可回到本书目录页 按键盘上方向键 ↑ 可回到本页顶部!
温馨提示: 温看小说的同时发表评论,说出自己的看法和其它小伙伴们分享也不错哦!发表书评还可以获得积分和经验奖励,认真写原创书评 被采纳为精评可以获得大量金币、积分和经验奖励哦!