`

CWnd中PreCreateWindow、PreSubclassWindow、SubclassWindow的区别

    博客分类:
  • VC
阅读更多

CWnd中PreCreateWindow、PreSubclassWindow、SubclassWindow的区别

http://blog.csdn.net/swimmer2000/archive/2007/10/30/1856213.aspx 


MFC(VC6.0)的CWnd及其子类中,有如下三个函数:

class CWnd : public CCmdTarget
{
public:    
    virtual BOOL PreCreateWindow(CREATESTRUCT& cs);
    virtual void PreSubclassWindow();
    BOOL SubclassWindow(HWND hWnd);    
};
 

  让人很不容易区分,不知道它们究竟干了些什么,在什么情况下要改写哪个函数?

  想知道改写函数?让我先告诉你哪个不能改写,那就是SubclassWindow。Scott Meyers的杰作<<Effective C++>>的第36条是这样的Differentiate between inheritance of interface and inheritance of implementation. 看了后你马上就知道,父类中的非虚拟函数是设计成不被子类改写的。根据有无virtual关键字,我们在排除了SubclassWindow后,也就知道PreCreateWindow和PreSubClassWindow是被设计成可改写的。接着的问题便是该在什么时候该写了。要知道什么时候该写,必须知道函数是在什么时候被调用,还有执行函数的想要达到的目的。我们先看看对这三个函数,MSDN给的解释:

  PreCreateWindow:

  Called by the framework before the creation of the Windows window 

  attached to this CWnd object.

  (译:在窗口被创建并attach到this指针所指的CWnd对象之前,被framework调用)

  PreSubclassWindow:

  This member function is called by the framework to allow other necessary 

  subclassing to occur before the window is subclassed.

  (译:在window被subclassed之前被framework调用,用来允许其它必要的subclassing发生)

虽然我已有译文,但还是让我对CWnd的attach和窗口的subclass作简单的解释吧!要理解attach,我们必须要知道一个C++的CWnd对象和窗口(window)的区别:window就是实在的窗口,而CWnd就是MFC用类对window所进行C++封装。attach,就是把窗口附加到CWnd对象上操作。附加(attach)完成后,CWnd对象才和窗口发生了联系。窗口的subclass是指修改窗口过程的操作,而不是面向对象中的派生子类。

  好了,PreCreateWindow由framework在窗口创建前被调用,函数名也说明了这一点,Pre应该是previous的缩写,PreSubclassWindow由framework在subclass窗口前调用。 这段话说了等于没说,你可能还是不知道,什么时候该改写哪个函数。罗罗嗦嗦的作者,还是用代码说话吧!源码之前,了无秘密(候捷语)。我们就看看MFC中的这三个函数都是这样实现的吧!

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先设定cs(CREATESTRUCT),在调用真正的窗口创建函数::CreateWindowEx之前,调用了CWnd::PreCreateWindow函数,并把参数cs以引用的方式传递了进去。而CWnd的PreCreateWindow函数也只是给cs.lpszClass赋值而已。毕竟,窗口创建函数CWnd::CreateEx的诸多参数中,并没有哪个指定了所要创建窗口的窗口类,而这又是不可缺少的(请参考<<windows程序设计>>第三章)。所以当你需要修改窗口的大小、风格、窗口所属的窗口类等cs成员变量时,要改写PreCreateWindow函数。

// 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先调用函数Attach(hWnd)让CWnd对象和hWnd所指的窗口发生关联。接着在用::SetWindowLong修改窗口过程(subclass)前,调用了PreSubclassWindow。CWnd::PreSubclassWindow则是什么都没有做。

  在CWnd的实现中,除了CWnd::SubclassWindow会调用PreSubclassWindow外,还有一处。上面所列函数CreateEx的代码,其中调用了一个AfxHookWindowCreate函数,见下面代码:

// 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函数用于设置一个挂钩函数(Hook函数)_AfxCbtFilterHook,每当Windows产生一个窗口时(还有许多其它类似,请参考<<深入浅出MFC>>第9章,563页),就会调用你设定的Hook函数。

  这样设定完成后,回到CWnd::CreateEx函数中,执行::CreateWindowEx进行窗口创建,窗口一产生,就会调用上面设定的Hook函数_AfxCbtFilterHook。而正是在_AfxCbtFilterHook中对函数PreSubclassWindow进行了第二次调用。见如下代码:

// 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;
    }
    
}
 

  也在调用函数SetWindowLong进行窗口subclass前调用了PreSubclassWindow.

posted on 2007-12-08 10:59 冷夜月 阅读(3010) 评论(1)  编辑 收藏 引用 所属分类: VC/MFC



评论

# re: [转]CWnd中PreCreateWindow、PreSubclassWindow、SubclassWindow的区别 2007-12-08 11:07 冷夜月

  通常情况下窗口是由用户创建的 CWnd::Create(..) 

  ●在此流程中,MFC提供一个机会"PreCreateWindow()供用户在创建前作点手脚 

  而对于对话框等,窗口是通过subclass方式交给用户的 

  系统读入对话框模板,建立其中各个子窗口 

  然后将各子窗口的 消息处理函数替换成 对应的C++对象 的消息处理函数 (Subclass:子类化,或"接管") ,然后,这个子窗口就会按类中定义的方式来动作了。 

  在此过程中,调用的是CWnd:SubclassWindow( HWND hWnd ); 

  ●在此流程中,MFC提供一个机会"PreSubclassWindow" 供用户在关联前作点手脚 

  具体来说,如果你定义一个窗口(如CButton派生类CMyButton),然后使用对话框数据交换将一个按钮与自己的派生类对象关联,这时候,一些"建立前"的处理就应该写在"PreSubclassWindow"中。 

  如果你用的不是"对话框数据关联",而是在OnInitDialg中自己创建m_mybtn.Create(...) 

  这时候,一些"建立前"的处理就应该写在 "PreCreateWindow"中。   回复  更多评论 

分享到:
评论

相关推荐

    gain-CWnd.rar_mfc中获取cwnd_visual c_获取HWND_获取窗口口cwnd

    标题"gain-CWnd.rar_mfc中获取cwnd_visual c_获取HWND_获取窗口口cwnd"涉及到的核心知识点是:如何在MFC应用中通过`HWND`来获取对应的`CWnd`对象。这在需要对窗口进行操作或消息处理时非常有用。下面将详细介绍这个...

    CWnd-MFC中文手册.pdf

    5. 窗口子类化与反子类化:子类化指的是通过重写窗口过程函数来改变窗口的行为,这里提及的PreSubclassWindow,SubclassWindow,UnsubclassWindow都与窗口子类化操作相关。 6. 窗口句柄与映射:例如CWndFromHandle...

    Cwnd全部函数[收集].pdf

    在软件开发领域,尤其是涉及到Windows应用程序编程时,`CWnd`类是Microsoft Foundation Classes (MFC)库中的一个核心组件。`CWnd`是MFC中用于与Windows窗口进行交互的基类,它封装了Windows API中的窗口句柄(HWND)...

    CWnd类成员函数详解

    在MFC(Microsoft Foundation Classes)中,`CWnd`类是所有窗口类的基础类,它提供了广泛的功能来管理和操作Windows窗口。本篇文章将深入解析`CWnd`类的部分关键成员函数,这些函数涵盖了窗口的创建、管理、样式修改、...

    CWnd窗口的全屏显示

    "CWnd窗口的全屏显示"是MFC编程中一个常见的需求,通常用于创建游戏、多媒体应用或演示文稿等场景。本篇文章将深入探讨如何使用MFC实现CWnd窗口的全屏显示。 首先,了解CWnd类在MFC中的角色至关重要。CWnd是MFC中的...

    怎样将CWnd转换为HWND和HWND转换为CWnd

    怎样将CWnd转换为HWND和HWND转换为CWnd

    CWnd类(VC++)所有函数的中文翻译

    在VC++编程环境中,`CWnd`类是MFC(Microsoft Foundation Classes)库中的核心类之一,它直接或间接地派生自`CObject`,并封装了Windows API中的`HWND`,即窗口句柄。`CWnd`类提供了一系列的方法来处理窗口创建、...

    从CWnd实现QQ好友控件

    在Windows编程领域,CWnd是MFC(Microsoft Foundation Classes)框架中的一个基类,它代表了一个窗口对象。从CWnd派生意味着创建一个新的窗口类,这个类可以有自己的特性、行为和外观。在这个特定的项目中,“从CWnd...

    SubclassWindow

    在MFC中,子类化窗口通常通过`CWnd`类的`SubclassWindow()`成员函数实现。这个函数用于将一个已存在的窗口与MFC的`CWnd`对象关联起来,从而让我们可以通过MFC的API来处理该窗口的消息。 1. **子类化过程**: - ...

    CWnd-MFC.rar_CWnd

    在Microsoft Foundation Class (MFC)库中,`CWnd`是一个非常重要的基类,它封装了Windows API中的窗口句柄(HWND)。`CWnd`类是MFC框架中所有窗口对象的基础,包括对话框、视图、框架窗口等。本资料“CWnd-MFC.rar”...

    如何通过HWND获得CWnd指针

    在MFC中,`CWnd`类是所有窗口对象的基础,它封装了Windows API中的`HWND`句柄。`HWND`是操作系统用来唯一标识窗口的整数值,而`CWnd`则提供了面向对象的接口,使得我们可以方便地处理窗口事件、绘制等内容。本文将...

    MFC CWnd重写并且加入控件动态创建,VS2005环境编译

    在Microsoft Foundation Classes (MFC)库中,`CWnd`是一个重要的基础类,它封装了Windows API中的`HWND`(窗口句柄)。`CWnd`是所有MFC窗口类(如`CButton`, `CEdit`, `CListBox`等)的基类,用于处理窗口消息和事件...

    CWnd 各类消息扩展

    在Windows编程中,`CWnd` 是MFC(Microsoft Foundation Classes)库中的一个核心类,它代表了Windows窗口。`CWnd` 类是所有用户界面控件和窗口类的基类,提供了处理窗口消息和事件的基本框架。在这个“CWnd 各类消息...

    CWnd自绘按钮

    在Windows编程中,CWnd是MFC(Microsoft Foundation Classes)框架中的一个基类,它代表了一个窗口对象。CWnd类提供了创建、管理以及与Windows API交互的基本功能。当我们需要对标准控件进行自定义,比如改变其外观...

    子窗体透明,CWnd继承

    标题"子窗体透明,CWnd继承"指出我们将探讨如何在MFC(Microsoft Foundation Classes)环境中,通过继承自CWnd类来实现子窗体的透明效果。描述中提到的链接指向一个可能的解决方案,同时提到了对VS2005的兼容性修复...

    继承CWnd重新写的CListCtrl

    1. **继承CWnd**:CWnd是MFC中所有窗口类的基础,它代表Windows操作系统中的一个窗口。通过继承CWnd,开发者可以获得创建和管理窗口的基本功能,并可以添加自己的方法和属性。在这个案例中,开发者可能通过重写或...

Global site tag (gtag.js) - Google Analytics