Chinaunix首页 | 论坛 | 博客
  • 博客访问: 367807
  • 博文数量: 56
  • 博客积分: 2721
  • 博客等级: 中校
  • 技术积分: 460
  • 用 户 组: 普通用户
  • 注册时间: 2008-07-29 16:18
文章分类

全部博文(56)

文章存档

2014年(1)

2011年(15)

2010年(5)

2008年(35)

分类: C/C++

2011-01-03 15:53:19

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

  1. // From VS Install PathVC98MFCIncludeAFXWIN.H
  2. class CWnd : public CCmdTarget
  3. {
  4.     ...
  5. public:
  6.     ...
  7.     virtual BOOL PreCreateWindow(CREATESTRUCT& cs);
  8.     virtual void PreSubclassWindow();
  9.     BOOL SubclassWindow(HWND hWnd);
  10.     ...
  11. };
  12. 让人很不容易区分,不知道它们究竟干了些什么,在什么情况下要改写哪个函数?

    想知道改写函数?让我先告诉你哪个不能改写,那就是SubclassWindow。Scott Meyers的杰作<>的第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中的这三个函数都是这样实现的吧! 

    1. // From VS Install PathVC98MFCSRCWINCORE.CPP
    2. BOOL CWnd::CreateEx(DWORD dwExStyle, LPCTSTR lpszClassName,
    3.                     LPCTSTR lpszWindowName, DWORD dwStyle,
    4.                     int x, int y, int nWidth, int nHeight,
    5.                     HWND hWndParent, HMENU nIDorHMenu, LPVOID lpParam)
    6. {
    7.     // allow modification of several common create parameters
    8.     CREATESTRUCT cs;
    9.     cs.dwExStyle = dwExStyle;
    10.     cs.lpszClass = lpszClassName;
    11.     cs.lpszName = lpszWindowName;
    12.     cs.style = dwStyle;
    13.     cs.x = x;
    14.     cs.y = y;
    15.     cs.cx = nWidth;
    16.     cs.cy = nHeight;
    17.     cs.hwndParent = hWndParent;
    18.     cs.hMenu = nIDorHMenu;
    19.     cs.hInstance = AfxGetInstanceHandle();
    20.     cs.lpCreateParams = lpParam;

    21.     if (!PreCreateWindow(cs))
    22.     {
    23.         PostNcDestroy();
    24.         return FALSE;
    25.     }

    26.     AfxHookWindowCreate(this);
    27.     HWND hWnd = ::CreateWindowEx(cs.dwExStyle, cs.lpszClass,
    28.     cs.lpszName, cs.style, cs.x, cs.y, cs.cx, cs.cy,
    29.     cs.hwndParent, cs.hMenu, cs.hInstance, cs.lpCreateParams);

    30.     ...
    31.     return TRUE;
    32. }

    33. // for child windows
    34. BOOL CWnd::PreCreateWindow(CREATESTRUCT& cs)
    35. {
    36.     if (cs.lpszClass == NULL)
    37.     {
    38.         // make sure the default window class is registered
    39.         VERIFY(AfxDeferRegisterClass(AFX_WND_REG));

    40.         // no WNDCLASS provided - use child window default
    41.         ASSERT(cs.style & WS_CHILD);
    42.         cs.lpszClass = _afxWnd;
    43.     }
    44.     return TRUE;
    45. }
    46. CWnd::CreateEx先设定cs(CREATESTRUCT),在调用真正的窗口创建函数::CreateWindowEx之前,调用了CWnd::PreCreateWindow函数,并把参数cs以引用的方式传递了进去。而CWnd的PreCreateWindow函数也只是给cs.lpszClass赋值而已。毕竟,窗口创建函数CWnd::CreateEx的诸多参数中,并没有哪个指定了所要创建窗口的窗口类,而这又是不可缺少的(请参考<>第三章)。所以当你需要修改窗口的大小、风格、窗口所属的窗口类等cs成员变量时,要改写PreCreateWindow函数。 

      1. // From VS Install PathVC98MFCSRCWINCORE.CPP
      2. BOOL CWnd::SubclassWindow(HWND hWnd)
      3. {
      4.     if (!Attach(hWnd))
      5.         return FALSE;

      6.     // allow any other subclassing to occur
      7.     PreSubclassWindow();

      8.     // now hook into the AFX WndProc
      9.     WNDPROC* lplpfn = GetSuperWndProcAddr();
      10.     WNDPROC oldWndProc = (WNDPROC)::SetWindowLong(hWnd, GWL_WNDPROC,
      11.             (DWORD)AfxGetAfxWndProc());
      12.     ASSERT(oldWndProc != (WNDPROC)AfxGetAfxWndProc());

      13.     if (*lplpfn == NULL)
      14.         *lplpfn = oldWndProc; // the first control of that type created
      15. #ifdef _DEBUG
      16.     else if (*lplpfn != oldWndProc)
      17.     {
      18.         ...
      19.         ::SetWindowLong(hWnd, GWL_WNDPROC, (DWORD)oldWndProc);
      20.     }
      21. #endif

      22.     return TRUE;
      23. }

      24. void CWnd::PreSubclassWindow()
      25. {
      26.     // no default processing
      27. }
      28. CWnd::SubclassWindow先调用函数Attach(hWnd)让CWnd对象和hWnd所指的窗口发生关联。接着在用::SetWindowLong修改窗口过程(subclass)前,调用了PreSubclassWindow。CWnd::PreSubclassWindow则是什么都没有做。

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

        1. // From VS Install PathVC98MFCSRCWINCORE.CPP
        2. BOOL CWnd::CreateEx(...)
        3. {
        4.     // allow modification of several common create parameters
        5.     ...

        6.     if (!PreCreateWindow(cs))
        7.     {
        8.         PostNcDestroy();
        9.         return FALSE;
        10.     }

        11.     AfxHookWindowCreate(this);
        12.     HWND hWnd = ::CreateWindowEx(cs.dwExStyle, cs.lpszClass,
        13.             cs.lpszName, cs.style, cs.x, cs.y, cs.cx, cs.cy,
        14.             cs.hwndParent, cs.hMenu, cs.hInstance, cs.lpCreateParams);

        15.     ...
        16.     return TRUE;
        17. }
        18. 接着察看AfxHookWindowCreate的代码:

          1. // From VS Install PathVC98MFCSRCWINCORE.CPP
          2. void AFXAPI AfxHookWindowCreate(CWnd* pWnd)
          3. {
          4.     ...

          5.     if (pThreadState->m_hHookOldCbtFilter == NULL)
          6.     {
          7.         pThreadState->m_hHookOldCbtFilter = ::SetWindowsHookEx(WH_CBT,
          8.                       _AfxCbtFilterHook, NULL, ::GetCurrentThreadId());
          9.         if (pThreadState->m_hHookOldCbtFilter == NULL)
          10.             AfxThrowMemoryException();
          11.     }
          12.     ...
          13. }
          14. 其主要作用的::SetWindowsHookEx函数用于设置一个挂钩函数(Hook函数)_AfxCbtFilterHook,每当Windows产生一个窗口时(还有许多其它类似,请参考<<深入浅出MFC>>第9章,563页),就会调用你设定的Hook函数。

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

            1. // From VS Install PathVC98MFCSRCWINCORE.CPP
            2. /////////////////////////////////////////////////////////////////////////////
            3. // Window creation hooks

            4. LRESULT CALLBACK
            5. _AfxCbtFilterHook(int code, WPARAM wParam, LPARAM lParam)
            6. {
            7.     ...
            8.     // connect the HWND to pWndInit...
            9.     pWndInit->Attach(hWnd);
            10.     // allow other subclassing to occur first
            11.     pWndInit->PreSubclassWindow();
            12.     ...
            13.     {
            14.         // subclass the window with standard AfxWndProc
            15.         oldWndProc = (WNDPROC)SetWindowLong(hWnd, GWL_WNDPROC, (DWORD)afxWndProc);
            16.         ASSERT(oldWndProc != NULL);
            17.         *pOldWndProc = oldWndProc;
            18.     }
            19.     ...
            20. }

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

阅读(1499) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~