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

深入浅出MFC第2版(PDF格式)-第73部分

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


                                               7 

           CWinApp* pApp = AfxGetApp();          m_pMainWnd…》UpdateWindow(); 

                                               8 

                                                 return TRUE; 

         2                                    } 

           AfxWinInit(。。。); 



         3                                    CMyFrameWnd::CMyFrameWnd() 

           pApp…》InitApplication(); 

           pApp…》InitInstance();              { 

         4 

                                               6 

           nReturnCode = pApp…》Run();            Create(NULL; 〃Hello MFC〃; 。。。; 

                                                        〃MainMenu〃); 

           AfxWinTerm();                      } 

        } 

                                              void CMyFrameWnd::OnPaint()  { 。。。 } 

                                              void CMyFrameWnd::OnAbout()  { 。。。 } 



                                              BEGIN_MESSAGE_MAP(CMyFrameWnd; CFrameWnd) 

                                                 ON_MAND(IDM_ABOUT; OnAbout) 

                                                 ON_WM_PAINT() 

                                              END_MESSAGE_MAP() 



       CMyFrameWnd::CMyFrameWnd 结束后, 窗口已经诞生出来; 程序流程又回到 



       CMyWinApp::InitInstance  , 于是调用ShowWindow  函数令窗口显示出来, 并调用 



       UpdateWindow 函数令Hello 程序送出WM_PAINT  消息。 



       我们很关心这个WM_PAINT  消息如何送到窗口函数的手中。而且,窗口函数又在哪里? 



       MFC 程序是不是也像SDK 程序一样,有一个GetMessage/DispatchMesage 循环?是否 



       每个窗口也都有一个窗口函数,并以某种方式进行消息的判断与处理? 



       两者都是肯定的。我们马上来寻找证据。 



                                                                                               389 


…………………………………………………………Page 452……………………………………………………………

                       第篇    湷觥 FC  程式設計 



              CWinApp::Run 程序生命的活水源头 



                                                                  HELLO。CPP 



                                                               1  CMyWinApp theApp;   // application object 

                       WINMAIN。CPP 



                                                                  BOOL CMyWinApp::InitInstance() 

                        int AFXAPI AfxWinMain (。。。) 

                                                                   { 

                        { 

                                                                   5 

                                                                     m_pMainWnd = new CMyFrameWnd(); 

                            CWinApp* pApp = AfxGetApp(); 

                                                                     m_pMainWnd…》ShowWindow(m_nCmdShow); 

                                                                   7 

                                                                     m_pMainWnd…》UpdateWindow(); 

                         2                                         8 

                            AfxWinInit(。。。); 

                                                                     return TRUE; 

                         3                                         } 

                            pApp…》InitApplication(); 

                            pApp…》InitInstance(); 

                         4 

                                                                  CMyFrameWnd::CMyFrameWnd() 

                            nReturnCode = pApp…》Run(); 

                         9                                         { 

                                                                   6 

                                                                     Create(NULL; 〃Hello MFC〃; 。。。; 

                            AfxWinTerm(); 

                                                                            〃MainMenu〃); 

                        } 

                                   CWinApp::Run                    } 

                                    CWinApp::Run 



                            CWinThread::Run                       void CMyFrameWnd::OnPaint()  { 。。。 } 

                            do {                                  void CMyFrameWnd::OnAbout()  { 。。。 } 

                             do { 

                              ::GetMessage(&msg;。。。); 

                               ::GetMessage(&msg;。。。); 

                              PreTranslateMessage{&msg); 

                               PreTranslateMessage{&msg);         BEGIN_MESSAGE_MAP(CMyFrameWnd; CFrameWnd) 

                              ::TranslateMessage(&msg); 

                               ::TranslateMessage(&msg); 

                               ::DispatchMessage(&msg);              ON_MAND(IDM_ABOUT; OnAbout) 

                                ::DispatchMessage(&msg); 

                              。。。                                    ON_WM_PAINT() 

                               。。。 

                            } while (::PeekMessage(。。。));         END_MESSAGE_MAP() 

                             } while (::PeekMessage(。。。)); 



                                                 DefWindowProc 

                                                 DefWindowProc                              AfxWndProc 

                                                                   hooking and subclassing   AfxWndProc 



                                                                   (please see chap9; p。560; 

                                                                  “Minotauros and Theseus”) 



                     Hello 程序进行到这里,窗口类别注册好了,窗口诞生并显示出来了,UpdateWindow 被 



                     调用,使得消息队列中出现了一个WM_PAINT 消息,等待被处理。现在,执行的脚步到 



                     达pApp …》Run 。 



                     稍早我说过了,pApp  指向CMyWinApp 对象(也就是本例的theApp ),所以,当程序 



                     调用: 



                        pApp…》Run(); 



                     相当于调用: 



                        CMyWinApp::Run(); 



390 


…………………………………………………………Page 453……………………………………………………………

                                                          第6章    MFC 程式的生死因果 



要知道,CMyWinApp 继承自CWinApp,而Run 又是CWinApp 的一个虚拟函数。我们 



并没有改写它(大部份情况下不需改写它),所以上述动作相当于调用: 



   CWinApp::Run(); 



其源代码出现在APPCORE。CPP  中: 



int CWinApp::Run() 

 { 

    if (m_pMainWnd == NULL && AfxOleGetUserCtrl()) 

    { 

        // Not launched /Embedding or /Automation; but has no main window! 

        TRACE0(〃Warning: m_pMainWnd is NULL in CWinApp::Run quitting 

         application。n〃); 

        AfxPostQuitMessage(0); 

    } 

    return CWinThread::Run(); 

 } 



32 位MFC 与16 位MFC  的巨大差异在于CWinApp 与CCmdTarget 之间多出了一 



个CWinThread,事情变得稍微复杂一些。CWinThread 定义于THRDCORE。CPP : 



int CWinThread::Run() 

 { 

    // for tracking the idle time state 

    BOOL bIdle = TRUE; 

    LONG lIdleCount = 0; 



    // acquire and dispatch messages until a WM_QUIT message is received。 

    for (;;) 

    { 

        // phase1: check to see if we can do idle work 

        while (bIdle && 

                !::PeekMessage(&m_msgCur; NULL; NULL; NULL; PM_NOREMOVE)) 

        { 

            // call OnIdle while in bIdle state 

            if (!OnIdle(lIdleCount++)) 

                bIdle = FALSE; // assume 〃no idle〃 state 

        } 



        // phase2: pump messages while available 

        do 

        { 

            // pump message; but quit on WM_QUIT 



                                                                                           391 


…………………………………………………………Page 454……………………………………………………………

                    第篇    湷觥 FC  程式設計 



                                if (!PumpMessage ()) 

                                    return ExitInstance(); 



                                // reset 〃no idle〃 state after pumping 〃normal〃 message 

                                if (IsIdleMessage(&m_msgCur)) 

                                { 

                                    bIdle = TRUE; 

                                    lIdleCount = 0; 

                                } 



                            } while (::PeekMessage(&m_msgCur; NULL; NULL; NULL; PM_NOREMOVE)); 

                        } 



                        ASSERT(FALSE);  // not reachable 

                     } 



                     BOOL CWinThread::PumpMessage() 

                     { 

                        if (!::GetMessage(&m_msgCur; NULL; NULL; NULL)) 

                        { 

                            return FALSE; 

                        } 



                        // process this message 

                        if (m_msgCur。message != WM_KICKIDLE && !PreTranslateMessage(&m_msgCur)) 

                        { 

                            ::TranslateMessage(&m_msgCur); 

                            ::DispatchMessage(&m_msgCur); 

                        } 

                        return TRUE; 

                     } 



                      获得的消息如何交给适当的例程去处理呢?SDK 程序的作法是调用DispatchMessage ,把 



                      消息丢给窗口函数;MFC 也是如此。但我们并未在Hello 程序中提供任何窗口函数,是 



                      的,窗口函数事实上由MFC 提供。回头看看前面AfxEndDeferRegisterClass 源代码,它 



                      在注册四种窗口类别之前已经指定窗口函数为: 



                        wndcls。lpfnWndProc = DefWindowProc; 



392 


…………………………………………………………Page 455……………………………………………………………

                                         第6章    MFC 程式的生死因果 



注意,虽然窗口函数被指定为DefWindowProc 成员函数,但事实上消息并不是被唧往该 



处,而是一个名为AfxWndProc  的全域函数去。这其中牵扯到MFC 暗中做了大挪移的 



手脚(利用hook 和subclassing),我将在第9章详细讨论这个「乾坤大挪移」。 



你看,WinMain 已由MFC 提供,窗口类别已由MFC 注册完成、连窗口函数也都由MFC 



提供。那么我们(程序员)如何为特定的消息设计特定的处理例程?MFC 应用程序对讯 



息的辨识与判别是采用所谓的「Message Map 机制」。 



                                                                393 


…………………………………………………………Page 456……………………………………………………………

                            第篇    湷觥 FC  程式設計 



                  把消息与处理函数串接在一起:Message Map 机制 



                                                                                     HELLO。CPP 



                                                                                1    CMyWinApp theApp;   // application object 



                                                                                     BOOL CMyWinApp::InitInstance() 

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