• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            笑看風(fēng)云淡

            寵辱不驚,看庭前花開(kāi)花落;去留無(wú)意,望天空云卷云舒
            posts - 96, comments - 48, trackbacks - 0, articles - 0
              C++博客 :: 首頁(yè) :: 新隨筆 ::  :: 聚合  :: 管理
            MFC(VC6.0)的CWnd及其子類(lèi)中,有如下三個(gè)函數(shù):
            class CWnd : public CCmdTarget
            {
                
            public:
                
                    
            virtual BOOL PreCreateWindow(CREATESTRUCT& cs);
                
            virtual void PreSubclassWindow();
                BOOL SubclassWindow(HWND hWnd);
                
            }
            ;
              讓人很不容易區(qū)分,不知道它們究竟干了些什么,在什么情況下要改寫(xiě)哪個(gè)函數(shù)?
              想知道改寫(xiě)函數(shù)?讓我先告訴你哪個(gè)不能改寫(xiě),那就是SubclassWindow。Scott Meyers的杰作<<Effective C++>>的第36條是這樣的Differentiate between inheritance of interface and inheritance of implementation. 看了后你馬上就知道,父類(lèi)中的非虛擬函數(shù)是設(shè)計(jì)成不被子類(lèi)改寫(xiě)的。根據(jù)有無(wú)virtual關(guān)鍵字,我們?cè)谂懦薙ubclassWindow后,也就知道PreCreateWindow和PreSubClassWindow是被設(shè)計(jì)成可改寫(xiě)的。接著的問(wèn)題便是該在什么時(shí)候該寫(xiě)了。要知道什么時(shí)候該寫(xiě),必須知道函數(shù)是在什么時(shí)候被調(diào)用,還有執(zhí)行函數(shù)的想要達(dá)到的目的。我們先看看對(duì)這三個(gè)函數(shù),MSDN給的解釋:
              PreCreateWindow:
              Called by the framework before the creation of the Windows window
              attached to this CWnd object.
              (譯:在窗口被創(chuàng)建并attach到this指針?biāo)傅腃Wnd對(duì)象之前,被framework調(diào)用)
              PreSubclassWindow:
              This member function is called by the framework to allow other necessary
              subclassing to occur before the window is subclassed.
              (譯:在window被subclassed之前被framework調(diào)用,用來(lái)允許其它必要的subclassing發(fā)生)
            雖然我已有譯文,但還是讓我對(duì)CWnd的attach和窗口的subclass作簡(jiǎn)單的解釋吧!要理解attach,我們必須要知道一個(gè)C++的CWnd對(duì)象和窗口(window)的區(qū)別:window就是實(shí)在的窗口,而CWnd就是MFC用類(lèi)對(duì)window所進(jìn)行C++封裝。attach,就是把窗口附加到CWnd對(duì)象上操作。附加(attach)完成后,CWnd對(duì)象才和窗口發(fā)生了聯(lián)系。窗口的subclass是指修改窗口過(guò)程的操作,而不是面向?qū)ο笾械呐缮宇?lèi)。
              好了,PreCreateWindow由framework在窗口創(chuàng)建前被調(diào)用,函數(shù)名也說(shuō)明了這一點(diǎn),Pre應(yīng)該是previous的縮寫(xiě),PreSubclassWindow由framework在subclass窗口前調(diào)用。 這段話說(shuō)了等于沒(méi)說(shuō),你可能還是不知道,什么時(shí)候該改寫(xiě)哪個(gè)函數(shù)。羅羅嗦嗦的作者,還是用代碼說(shuō)話吧!源碼之前,了無(wú)秘密(候捷語(yǔ))。我們就看看MFC中的這三個(gè)函數(shù)都是這樣實(shí)現(xiàn)的吧!
            BOOL CWnd::CreateEx(DWORD dwExStyle, LPCTSTR lpszClassName,
                                LPCTSTR lpszWindowName, DWORD dwStyle,
                                
            int x, int y, int nWidth, int nHeight,
                                HWND hWndParent, HMENU nIDorHMenu, LPVOID lpParam)
                                
            {
                
            // allow modification of several common create parameters
                CREATESTRUCT cs;
                cs.dwExStyle 
            = dwExStyle;
                cs.lpszClass 
            = lpszClassName;
                cs.lpszName 
            = lpszWindowName;
                cs.style 
            = dwStyle;
                cs.x 
            = x;
                cs.y 
            = y;
                cs.cx 
            = nWidth;
                cs.cy 
            = nHeight;
                cs.hwndParent 
            = hWndParent;
                cs.hMenu 
            = nIDorHMenu;
                cs.hInstance 
            = AfxGetInstanceHandle();
                cs.lpCreateParams 
            = lpParam;
                
                
            if (!PreCreateWindow(cs))
                    
            {
                    PostNcDestroy();
                    
            return FALSE;
                }

                
                AfxHookWindowCreate(
            this);
                HWND hWnd 
            = ::CreateWindowEx(cs.dwExStyle, cs.lpszClass,
                    cs.lpszName, cs.style, cs.x, cs.y, cs.cx, cs.cy,
                    cs.hwndParent, cs.hMenu, cs.hInstance, cs.lpCreateParams);
                
                
                    
            return TRUE;
            }


            // for child windows
            BOOL CWnd::PreCreateWindow(CREATESTRUCT& cs)
            {
                
            if (cs.lpszClass == NULL)
                    
            {
                    
            // make sure the default window class is registered
                    VERIFY(AfxDeferRegisterClass(AFX_WND_REG));
                    
                    
            // no WNDCLASS provided - use child window default
                    ASSERT(cs.style & WS_CHILD);
                    cs.lpszClass 
            = _afxWnd;
                }

                
            return TRUE;
            }
              CWnd::CreateEx先設(shè)定cs(CREATESTRUCT),在調(diào)用真正的窗口創(chuàng)建函數(shù)::CreateWindowEx之前,調(diào)用了CWnd::PreCreateWindow函數(shù),并把參數(shù)cs以引用的方式傳遞了進(jìn)去。而CWnd的PreCreateWindow函數(shù)也只是給cs.lpszClass賦值而已。畢竟,窗口創(chuàng)建函數(shù)CWnd::CreateEx的諸多參數(shù)中,并沒(méi)有哪個(gè)指定了所要?jiǎng)?chuàng)建窗口的窗口類(lèi),而這又是不可缺少的(請(qǐng)參考<<windows程序設(shè)計(jì)>>第三章)。所以當(dāng)你需要修改窗口的大小、風(fēng)格、窗口所屬的窗口類(lèi)等cs成員變量時(shí),要改寫(xiě)PreCreateWindow函數(shù)。
            // From VS Install PathVC98MFCSRCWINCORE.CPP
            BOOL CWnd::SubclassWindow(HWND hWnd)
            {
                
            if (!Attach(hWnd))
                    
            return FALSE;
                
                
            // allow any other subclassing to occur
                PreSubclassWindow();
                
                
            // now hook into the AFX WndProc
                WNDPROC* lplpfn = GetSuperWndProcAddr();
                WNDPROC oldWndProc 
            = (WNDPROC)::SetWindowLong(hWnd, GWL_WNDPROC,
                    (DWORD)AfxGetAfxWndProc());
                ASSERT(oldWndProc 
            != (WNDPROC)AfxGetAfxWndProc());
                
                
            if (*lplpfn == NULL)
                    
            *lplpfn = oldWndProc;   // the first control of that type created
            #ifdef _DEBUG
                
            else if (*lplpfn != oldWndProc)
                    
            {
                    
                        ::SetWindowLong(hWnd, GWL_WNDPROC, (DWORD)oldWndProc);
                }

            #endif
                
                
            return TRUE;
            }


            void CWnd::PreSubclassWindow()
            {
                
            // no default processing
            }
              CWnd::SubclassWindow先調(diào)用函數(shù)Attach(hWnd)讓CWnd對(duì)象和hWnd所指的窗口發(fā)生關(guān)聯(lián)。接著在用::SetWindowLong修改窗口過(guò)程(subclass)前,調(diào)用了PreSubclassWindow。CWnd::PreSubclassWindow則是什么都沒(méi)有做。
              在CWnd的實(shí)現(xiàn)中,除了CWnd::SubclassWindow會(huì)調(diào)用PreSubclassWindow外,還有一處。上面所列函數(shù)CreateEx的代碼,其中調(diào)用了一個(gè)AfxHookWindowCreate函數(shù),見(jiàn)下面代碼:
            // From VS Install PathVC98MFCSRCWINCORE.CPP
            BOOL CWnd::CreateEx()
            {
                
            // allow modification of several common create parameters
                
                    
                    
            if (!PreCreateWindow(cs))
                        
            {
                        PostNcDestroy();
                        
            return FALSE;
                    }

                    
                    AfxHookWindowCreate(
            this); 
                    HWND hWnd 
            = ::CreateWindowEx(cs.dwExStyle, cs.lpszClass,
                        cs.lpszName, cs.style, cs.x, cs.y, cs.cx, cs.cy,
                        cs.hwndParent, cs.hMenu, cs.hInstance, cs.lpCreateParams);
                    
                    
                        
            return TRUE;
            }
              接著察看AfxHookWindowCreate的代碼:

            // From VS Install PathVC98MFCSRCWINCORE.CPP
            void AFXAPI AfxHookWindowCreate(CWnd* pWnd)
            {
                
                    
                    
            if (pThreadState->m_hHookOldCbtFilter == NULL)
                        
            {
                        pThreadState
            ->m_hHookOldCbtFilter = ::SetWindowsHookEx(WH_CBT,
                            _AfxCbtFilterHook, NULL, ::GetCurrentThreadId());
                        
            if (pThreadState->m_hHookOldCbtFilter == NULL)
                            AfxThrowMemoryException();
                    }

                    
            }

              其主要作用的::SetWindowsHookEx函數(shù)用于設(shè)置一個(gè)掛鉤函數(shù)(Hook函數(shù))_AfxCbtFilterHook,每當(dāng)Windows產(chǎn)生一個(gè)窗口時(shí)(還有許多其它類(lèi)似,請(qǐng)參考<<深入淺出MFC>>第9章,563頁(yè)),就會(huì)調(diào)用你設(shè)定的Hook函數(shù)。
              這樣設(shè)定完成后,回到CWnd::CreateEx函數(shù)中,執(zhí)行::CreateWindowEx進(jìn)行窗口創(chuàng)建,窗口一產(chǎn)生,就會(huì)調(diào)用上面設(shè)定的Hook函數(shù)_AfxCbtFilterHook。而正是在_AfxCbtFilterHook中對(duì)函數(shù)PreSubclassWindow進(jìn)行了第二次調(diào)用。見(jiàn)如下代碼:
            // From VS Install PathVC98MFCSRCWINCORE.CPP
            /**//////////////////////////////////////////////////////////////////////////////
            // Window creation hooks

            LRESULT CALLBACK
            _AfxCbtFilterHook(
            int code, WPARAM wParam, LPARAM lParam)
            {
                       
                    
                    
            // connect the HWND to pWndInit
                    pWndInit->Attach(hWnd);
                
            // allow other subclassing to occur first
                pWndInit->PreSubclassWindow();
                
                    
            {
                    
            // subclass the window with standard AfxWndProc
                    oldWndProc = (WNDPROC)SetWindowLong(hWnd, GWL_WNDPROC, (DWORD)afxWndProc);
                    ASSERT(oldWndProc 
            != NULL);
                    
            *pOldWndProc = oldWndProc;
                }

                
            }
              也在調(diào)用函數(shù)SetWindowLong進(jìn)行窗口subclass前調(diào)用了PreSubclassWindow.

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

            通常情況下窗口是由用戶(hù)創(chuàng)建的
            CWnd::Create(..)
            ●在此流程中,MFC提供一個(gè)機(jī)會(huì)"PreCreateWindow()供用戶(hù)在創(chuàng)建前作點(diǎn)手腳

            而對(duì)于對(duì)話框等,窗口是通過(guò)subclass方式交給用戶(hù)的
            系統(tǒng)讀入對(duì)話框模板,建立其中各個(gè)子窗口

            然后將各子窗口的 消息處理函數(shù)替換成 對(duì)應(yīng)的C++對(duì)象 的消息處理函數(shù) (Subclass:子類(lèi)化,或"接管") ,然后,這個(gè)子窗口就會(huì)按類(lèi)中定義的方式來(lái)動(dòng)作了。

            在此過(guò)程中,調(diào)用的是CWnd:SubclassWindow( HWND hWnd );
            ●在此流程中,MFC提供一個(gè)機(jī)會(huì)"PreSubclassWindow" 供用戶(hù)在關(guān)聯(lián)前作點(diǎn)手腳

            具體來(lái)說(shuō),如果你定義一個(gè)窗口(如CButton派生類(lèi)CMyButton),然后使用對(duì)話框數(shù)據(jù)交換將一個(gè)按鈕與自己的派生類(lèi)對(duì)象關(guān)聯(lián),這時(shí)候,一些"建立前"的處理就應(yīng)該寫(xiě)在"PreSubclassWindow"中。

            如果你用的不是"對(duì)話框數(shù)據(jù)關(guān)聯(lián)",而是在OnInitDialg中自己創(chuàng)建m_mybtn.Create(...)
            這時(shí)候,一些"建立前"的處理就應(yīng)該寫(xiě)在
            "PreCreateWindow"中。

            這里“建立前”的處理包括像那些處理,跟PreCreateWindows()做的一些窗口初始化的工作有什么不同?

            PreCreateWindows函數(shù)中沒(méi)有窗口可以用——還沒(méi)有創(chuàng)建
            PreSubclassWindow函數(shù)中可以對(duì)窗口進(jìn)行操作。


            只有注冊(cè)用戶(hù)登錄后才能發(fā)表評(píng)論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問(wèn)   Chat2DB   管理


            国产无套内射久久久国产| 久久久久高潮毛片免费全部播放| 久久久久亚洲AV无码网站| 久久丫精品国产亚洲av不卡| 精品久久无码中文字幕| 久久精品无码一区二区三区日韩| 色播久久人人爽人人爽人人片aV| 国产精品久久久香蕉| 成人久久精品一区二区三区 | 久久涩综合| 久久人人爽人人爽人人片AV高清| 久久久久亚洲AV片无码下载蜜桃| 色综合久久综合网观看| 欧美激情一区二区久久久| 91精品国产91热久久久久福利| 久久亚洲精品无码观看不卡| 狠狠色婷婷久久一区二区三区| 久久久久人妻精品一区三寸蜜桃| 久久精品国产亚洲AV无码娇色| 日韩精品久久久久久| 亚洲综合熟女久久久30p| 久久福利片| 久久er国产精品免费观看2| 亚洲人成网站999久久久综合 | 精品久久久中文字幕人妻| 久久国产高清字幕中文| av色综合久久天堂av色综合在| 国产精品伦理久久久久久| 久久精品亚洲精品国产色婷| 欧美日韩精品久久免费| 亚州日韩精品专区久久久| 国内精品久久久久久久亚洲| 久久国产亚洲精品麻豆| 久久国产精品一国产精品金尊| 一本一本久久aa综合精品| 亚洲а∨天堂久久精品9966| 99久久无码一区人妻| 青青草原综合久久大伊人精品| 国产一区二区三区久久| 久久精品9988| 99久久精品国产毛片|