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

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

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




               這是個隨時等待接受訊息的小程式。它所接受的訊息,夾帶著  global atom  作為參數。 



               Tracewin  只要把  atom  解碼,丟到個由它管轄的Edit  視窗即可。Paul DiLascia  設計 



               的這個小程式,功能面面俱到,可以把接受的  TRACE  輸出字串顯示在視窗,或放到 



               某個檔案;也可以把  EDIT  緩衝區內容拷貝到剪貼簿,或是清除整個  EDIT  緩衝區 



               內容。功能與  Visual C++ 1。5  的DBWIN  幾乎不相,只差洠軌虬殉e字串輸出到 



               1  和  2  。 



               Tracewin  並不動用  Document/View  架構。主視窗內含個  Edit  控制元件作為其子視 



               窗,事實,那是個衍生自  CEdit  的  CBufWnd  類別,有點像  CEditView 。 



               當應用程式以  TRACE  巨集送出字串,經過  CDumpContext::OutputString  的作用,送往 



               CMfxTrace::Write          FindWindow    Tracewin  

                            ,我們在其以              找到        視窗: 



                  CWnd *pTraceWnd = CWnd::FindWindow(TRACEWND_CLASSNAME; NULL); 



               要知道,如果  Tracewin  使用的類別是  MFC  預先建立的個類別,那麼它的類別名稱 



               可能是像  Afx:b:14ae:6:3e8f  這種不太有意義的字串,而且可能在不同的機器不同的時間 



               有不同的名稱。如此來如何為  FindWindow  指定第個參數?我們必須有個什麼方法 



               避免使用  MFC  預先建立的個類別,但又能夠使用其類別設定。 



          視窗類別名稱  Afx:x:y:z:w 



               MFC 2。5  在應用程式開始執行時,便在  AfxWinInit  先行裕粤耍祩視窗類別備用。 



               MFC  4。x  的行為稍有修改,它在應用程式呼叫  LoadFrame  準備產生視窗時,才在 



               LoadFrame  所呼叫的  PreCreateWindow  虛擬函式為你產生視窗類別。5個可能的視窗 



               類別分別是: 



928 


…………………………………………………………Page 991……………………………………………………………

                                                    附錄D    以MFC 重建DBWIN  



    const TCHAR _afxWnd'' = AFX_WND; 

    const TCHAR _afxWndControlBar'' = AFX_WNDCONTROLBAR; 

    const TCHAR _afxWndMDIFrame'' = AFX_WNDMDIFRAME; 

    const TCHAR _afxWndFrameOrView'' = AFX_WNDFRAMEORVIEW; 

    const TCHAR _afxWndOleControl'' = AFX_WNDOLECONTROL; 



這些  AFX_xxx  常數定義於  AFXIMPL。H  ,依不同的聯結狀態(除錯模式與否、動態 



聯結與否)而有不同的值。如果使用  MFC  動態聯結版和除錯版,5個視窗類別的名稱 



將是: 



    〃AfxWnd42d〃 

    〃AfxControlBar42d〃 

    〃AfxMDIFrame42d〃 

    〃AfxFrameOrView42d〃 

    〃AfxOleControl42d〃 



如果是使用  MFC  靜態聯結版和除錯版,5個視窗類別的名稱將是: 



    〃AfxWnd42sd〃 

    〃AfxControlBar42sd〃 

    〃AfxMDIFrame42sd〃 

    〃AfxFrameOrView42sd〃 

    〃AfxOleControl42sd〃 



                                         深入湷錾钊霚出 MFC            /  

這些名稱的由來,以及它們的裕詴r機,請參考深入湷錾钊霚出                           (侯俊傑 松崗)第6 



章。 



然而,這些視窗類別名稱又怎麼會變成像  Afx:b:14ae:6:3e8f  這副奇怪模樣呢?原來是 



Framework  玩的把戲,它把這些視窗類別名稱轉換為  Afx:x:y:z:w  型式,成為獨無之 



視窗類別名稱: 



    x:  視窗風格(window style)的 hex 值 

    y:  滑鼠游標的 hex 值 

    z:  背景顏色的 hex 值 

    w:  圖示的 hex 值 



        CMfxTrace        FindWindow          Tracewin Tracewin  

為了讓              能夠以               函式找到             ,        的視窗類別名稱 



              Afx:x:y:z:w 

 (也就是那個                )必須完全在我們的控制之才行。那麼,我們勢必得改寫 



Tracewin  的  frame  視窗的  PreCreateWindow  虛擬函式。 



                                                                                 929 


…………………………………………………………Page 992……………………………………………………………

                   第五篇    附錄  



                   圖五      以   Spy   觀察視窗。請注意每一個視窗類別的名稱都是   Afx:x:y:z:w   型 

                           式。fig5。bmp 



             PreCreateWindow  和  GetClassInfo 



                   圖六是  Tracewin  的  PreCreateWindow  內容,利用  GetClassInfo  把  MFC  的視窗類別做 



                   出份副本,然後改變其類別名稱以及程式圖示,再重新裕浴J堑模匦略'冊是必要 



                                 CMfxTrace  Write         FindWindow 

                   的。這麼來,                 ::     呼叫的                 就有明確的搜尋目標了。 



                   #0001  BOOL CMainFrame::PreCreateWindow(CREATESTRUCT& cs) 

                   #0002  { 

                   #0003    static LPCSTR className = NULL; 

                   #0004 

                   #0005    if (!CFrameWnd::PreCreateWindow(cs)) 

                   #0006        return FALSE; 

                   #0007 

                   #0008    if (className==NULL) { 

                   #0009      // One…time class registration 

                   #0010      // The only purpose is to make the class name something 

                   #0011      // meaningful instead of 〃Afx:0x4d:27:32:hup1hup:hike!〃 

                   #0012      // 

                   #0013      WNDCLASS wndcls; 

                   #0014      ::GetClassInfo(AfxGetInstanceHandle(); cs。lpszClass; &wndcls); 

                   #0015      wndcls。lpszClassName = TRACEWND_CLASSNAME; 

                   #0016      wndcls。hIcon = AfxGetApp()…》LoadIcon(IDR_MAINFRAME); 

                   #0017      VERIFY(AfxRegisterClass(&wndcls)); 

                   #0018      className=TRACEWND_CLASSNAME; 



930 


…………………………………………………………Page 993……………………………………………………………

                                                                 附錄D    以MFC 重建DBWIN  



       #0019    } 

       #0020    cs。lpszClass = className; 

       #0021 

       #0022    // Load window position from profile 

       #0023    CWinApp *pApp = AfxGetApp(); 

       #0024    cs。x = pApp…》GetProfileInt(PROFILE; 〃x〃; CW_USEDEFAULT); 

       #0025    cs。y = pApp…》GetProfileInt(PROFILE; 〃y〃; CW_USEDEFAULT); 

       #0026    cs。cx = pApp…》GetProfileInt(PROFILE; 〃cx〃; CW_USEDEFAULT); 

       #0027    cs。cy = pApp…》GetProfileInt(PROFILE; 〃cy〃; CW_USEDEFAULT); 

       #0028 

       #0029    return TRUE; 

       #0030  } 



       圖六      Tracewin   的   frame   視窗的   PreCreateWindow   函式內容 



Tracewin  取出字串並顯示 



                                                      WM  TRACE  MSG            OnTraceMsg 

             Tracewin                                    _       _              

       如果             欲接收由除錯端傳來的自定訊息                                   ,並交由 



       函式去處理,它就必須在其訊息映射表有所表示: 



           BEGIN_MESSAGE_MAP(CMainFrame; CFrameWnd) 

             //{{AFX_MSG_MAP(CMainFrame) 

             ON_REGISTERED_MESSAGE(WM_TRACE_MSG; OnTraceMsg) 

             。。。 

             //}}AFX_MSG_MAP 

           END_MESSAGE_MAP() 



       並且設計  OnTraceMsg  函式如圖七。 



       #0001  LRESULT CMainFrame::OnTraceMsg(WPARAM wParam; LPARAM) 

       #0002  { 

       #0003      if (!wParam || m_nOutputWhere==ID_OUTPUT_OFF) 

       #0004              return 0; 

       #0005 

       #0006      char buf'256'; 

       #0007      UINT len = GlobalGetAtomName((ATOM)wParam; buf; sizeof(buf)); 

       #0008 

       #0009      if (m_nOutputWhere==ID_OUTPUT_TO_WINDOW) { 

       #0010 

       #0011          // Convert n to nr for Windows edit control 



                                                                                                  931 


…………………………………………………………Page 994……………………………………………………………

                  第五篇    附錄  



                  #0012          。。。 

                  #0013          // Append string to contents of trace buffer 

                  #0014          。。。 

                  #0015      } else if (m_nOutputWhere==ID_OUTPUT_TO_FILE) { 

                  #0016          m_file。Write(buf; len); 

                  #0017      } 

                  #0018      return 0; 

                  #0019  } 



                  圖七      CMainFrame::OnTraceMsg   函式內容。 



                    WM_COPYDATA                                      IPC 

            改用                                進行行程通訊(  ) 



                  Paul  DiLascia  的第篇文章發表後,收到許多讀者的來信,指出以  global  atom  完成行 



                                                    WM  COPYDATA 

                                                       _                  Paul  

                  程通訊並不是最高明的辦法,可以改用                                。於是       從善如流寫了第 



                  篇文章。 



                  WM  COPYDATA 

                     _            是  Win32  的新訊息,可以提供個簡單又方便的方法,把字串送往另 



                  個程式。這正符合  Tracewin  之所需。這個訊息的兩個參數意義如: 



                      wParam = (WPARAM) (HWND) wnd;             // handle of sending window 

                      lParam = (LPARAM) (PCOPYDATASTRUCT) pcds; // pointer to structure with data 



                  其  COPYDATASTRUCT  結構定義如: 



                      typedef struct tagCOPYDATASTRUCT {  // cds 

                          DWORD dwData; // 隨便你指定任何你需要的額外資訊 

                          DWORD cbData; // 資料長度 

                          PVOID lpData; // 資料指標 

                      } COPYDATASTRUCT; 



                  dwData                       ID_COPYDATA_TRACEMSG    Tracewin  

                         在本例應用被指定為                                   ;        將檢查這個識別 



                                               WM  COPYDATA 

                  碼,如果不合格,就忽略該次的                  _           訊息。 



                  Tracewin  新版本的內容我就不再列出了,請直接載其原始碼看個究竟。 



                



932 


…………………………………………………………Page 995……………………………………………………………

                                                           附錄D    以MFC 重建DBWIN  



我的使用經驗 



      現在讓我來談點我使用  Tracewin  的經驗。 



      我早就需要在  Visual  C++  使用  DBWIN  了,也早就看到了  Paul  DiLascia  的兩篇文 



      章,但是真正研讀它並使用其成果,是在我撰寫  /OLE/ActiveX  書(我最新的 



      本書,還在孵化之)的時候。也許當你讀到該書,會感嘆侯俊傑怎麼能夠對OLE container 



      和  server  之間的交叉動作瞭若指掌。洠в惺颤N,我只是在  container  和  server  之的每 



      個我感興趣的函式的開始處,利用  TRACE  輸出些訊息,這樣我就可以從容從 



      Tracewin  視窗觀察那些函式的被呼喚時機了。 



      所以我在  OLE container  這麼做: 



          #include 〃tracewin。h〃 

          。。。 

          BOOL CContainerApp::InitInstance() 

          { 

            。。。 

            pMainFrame…》ShowWindow(m_nCmdShow); 

            pMainFrame…》UpdateWindow(); 



            CMfxTrace::Init(); // add by J。J。Hou 

            return TRUE; 

          } 



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