[转]CWnd中PreCreateWindow、PreSubclassWindow、SubclassWindow的区别

http://blog.csdn.net/swimmer2000/archive/2007/10/30/1856213.aspx
  MFC(VC6.0)的CWnd及其子类中,有如下三个函数:
none.gif class CWnd: public CCmdTarget
expandedblockstart.gifcontractedblock.gifdot.gif
dot.gif {
inblock.gifdot.gif
inblock.gif
public:
inblock.gifdot.gif
inblock.gif
virtualBOOLPreCreateWindow(CREATESTRUCT&cs);
inblock.gif
virtualvoidPreSubclassWindow();
inblock.gifBOOLSubclassWindow(HWNDhWnd);
inblock.gifdot.gif
expandedblockend.gif}
;
  让人很不容易区分,不知道它们究竟干了些什么,在什么情况下要改写哪个函数?
  想知道改写函数?让我先告诉你哪个不能改写,那就是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中的这三个函数都是这样实现的吧!
none.gif BOOLCWnd::CreateEx(DWORDdwExStyle,LPCTSTRlpszClassName,
none.gifLPCTSTRlpszWindowName,DWORDdwStyle,
none.gif
int x, int y, int nWidth, int nHeight,
none.gifHWNDhWndParent,HMENUnIDorHMenu,LPVOIDlpParam)
expandedblockstart.gifcontractedblock.gifdot.gif
dot.gif {
inblock.gif
//allowmodificationofseveralcommoncreateparameters
inblock.gif
CREATESTRUCTcs;
inblock.gifcs.dwExStyle
=dwExStyle;
inblock.gifcs.lpszClass
=lpszClassName;
inblock.gifcs.lpszName
=lpszWindowName;
inblock.gifcs.style
=dwStyle;
inblock.gifcs.x
=x;
inblock.gifcs.y
=y;
inblock.gifcs.cx
=nWidth;
inblock.gifcs.cy
=nHeight;
inblock.gifcs.hwndParent
=hWndParent;
inblock.gifcs.hMenu
=nIDorHMenu;
inblock.gifcs.hInstance
=AfxGetInstanceHandle();
inblock.gifcs.lpCreateParams
=lpParam;
inblock.gif
inblock.gif
if(!PreCreateWindow(cs))
expandedsubblockstart.gifcontractedsubblock.gifdot.gif
dot.gif{
inblock.gifPostNcDestroy();
inblock.gif
returnFALSE;
expandedsubblockend.gif}

inblock.gif
inblock.gifAfxHookWindowCreate(
this);
inblock.gifHWNDhWnd
=::CreateWindowEx(cs.dwExStyle,cs.lpszClass,
inblock.gifcs.lpszName,cs.style,cs.x,cs.y,cs.cx,cs.cy,
inblock.gifcs.hwndParent,cs.hMenu,cs.hInstance,cs.lpCreateParams);
inblock.gif
inblock.gifdot.gif
inblock.gif
returnTRUE;
expandedblockend.gif}

none.gif
none.gif
// forchildwindows
none.gif
BOOLCWnd::PreCreateWindow(CREATESTRUCT & cs)
expandedblockstart.gifcontractedblock.gifdot.gif
dot.gif {
inblock.gif
if(cs.lpszClass==NULL)
expandedsubblockstart.gifcontractedsubblock.gifdot.gif
dot.gif{
inblock.gif
//makesurethedefaultwindowclassisregistered
inblock.gif
VERIFY(AfxDeferRegisterClass(AFX_WND_REG));
inblock.gif
inblock.gif
//noWNDCLASSprovided-usechildwindowdefault
inblock.gif
ASSERT(cs.style&WS_CHILD);
inblock.gifcs.lpszClass
=_afxWnd;
expandedsubblockend.gif}

inblock.gif
returnTRUE;
expandedblockend.gif}
  CWnd::CreateEx先设定cs(CREATESTRUCT),在调用真正的窗口创建函数::CreateWindowEx之前,调用了CWnd::PreCreateWindow函数,并把参数cs以引用的方式传递了进去。而CWnd的PreCreateWindow函数也只是给cs.lpszClass赋值而已。毕竟,窗口创建函数CWnd::CreateEx的诸多参数中,并没有哪个指定了所要创建窗口的窗口类,而这又是不可缺少的(请参考<<windows程序设计>>第三章)。所以当你需要修改窗口的大小、风格、窗口所属的窗口类等cs成员变量时,要改写PreCreateWindow函数。
none.gif // FromVSInstallPathVC98MFCSRCWINCORE.CPP
none.gif
BOOLCWnd::SubclassWindow(HWNDhWnd)
expandedblockstart.gifcontractedblock.gifdot.gif
dot.gif {
inblock.gif
if(!Attach(hWnd))
inblock.gif
returnFALSE;
inblock.gif
inblock.gif
//allowanyothersubclassingtooccur
inblock.gif
PreSubclassWindow();
inblock.gif
inblock.gif
//nowhookintotheAFXWndProc
inblock.gif
WNDPROC*lplpfn=GetSuperWndProcAddr();
inblock.gifWNDPROColdWndProc
=(WNDPROC)::SetWindowLong(hWnd,GWL_WNDPROC,
inblock.gif(DWORD)AfxGetAfxWndProc());
inblock.gifASSERT(oldWndProc
!=(WNDPROC)AfxGetAfxWndProc());
inblock.gif
inblock.gif
if(*lplpfn==NULL)
inblock.gif
*lplpfn=oldWndProc;//thefirstcontrolofthattypecreated
inblock.gif
#ifdef_DEBUG
inblock.gif
elseif(*lplpfn!=oldWndProc)
expandedsubblockstart.gifcontractedsubblock.gifdot.gif
dot.gif{
inblock.gifdot.gif
inblock.gif::SetWindowLong(hWnd,GWL_WNDPROC,(DWORD)oldWndProc);
expandedsubblockend.gif}

inblock.gif
#endif
inblock.gif
inblock.gif
returnTRUE;
expandedblockend.gif}

none.gif
none.gif
void CWnd::PreSubclassWindow()
expandedblockstart.gifcontractedblock.gifdot.gif
dot.gif {
inblock.gif
//nodefaultprocessing
expandedblockend.gif
}
  CWnd::SubclassWindow先调用函数Attach(hWnd)让CWnd对象和hWnd所指的窗口发生关联。接着在用::SetWindowLong修改窗口过程(subclass)前,调用了PreSubclassWindow。CWnd::PreSubclassWindow则是什么都没有做。
  在CWnd的实现中,除了CWnd::SubclassWindow会调用PreSubclassWindow外,还有一处。上面所列函数CreateEx的代码,其中调用了一个AfxHookWindowCreate函数,见下面代码:
none.gif // FromVSInstallPathVC98MFCSRCWINCORE.CPP
none.gif
BOOLCWnd::CreateEx(dot.gif)
expandedblockstart.gifcontractedblock.gifdot.gif
dot.gif {
inblock.gif
//allowmodificationofseveralcommoncreateparameters
inblock.gif
dot.gif
inblock.gif
inblock.gif
if(!PreCreateWindow(cs))
expandedsubblockstart.gifcontractedsubblock.gifdot.gif
dot.gif{
inblock.gifPostNcDestroy();
inblock.gif
returnFALSE;
expandedsubblockend.gif}

inblock.gif
inblock.gifAfxHookWindowCreate(
this);
inblock.gifHWNDhWnd
=::CreateWindowEx(cs.dwExStyle,cs.lpszClass,
inblock.gifcs.lpszName,cs.style,cs.x,cs.y,cs.cx,cs.cy,
inblock.gifcs.hwndParent,cs.hMenu,cs.hInstance,cs.lpCreateParams);
inblock.gif
inblock.gifdot.gif
inblock.gif
returnTRUE;
expandedblockend.gif}
  接着察看AfxHookWindowCreate的代码:
none.gif
none.gif
// FromVSInstallPathVC98MFCSRCWINCORE.CPP
none.gif
void AFXAPIAfxHookWindowCreate(CWnd * pWnd)
expandedblockstart.gifcontractedblock.gifdot.gif
dot.gif {
inblock.gifdot.gif
inblock.gif
inblock.gif
if(pThreadState->m_hHookOldCbtFilter==NULL)
expandedsubblockstart.gifcontractedsubblock.gifdot.gif
dot.gif{
inblock.gifpThreadState
->m_hHookOldCbtFilter=::SetWindowsHookEx(WH_CBT,
inblock.gif_AfxCbtFilterHook,NULL,::GetCurrentThreadId());
inblock.gif
if(pThreadState->m_hHookOldCbtFilter==NULL)
inblock.gifAfxThrowMemoryException();
expandedsubblockend.gif}

inblock.gifdot.gif
expandedblockend.gif}

none.gif
  其主要作用的::SetWindowsHookEx函数用于设置一个挂钩函数(Hook函数)_AfxCbtFilterHook,每当Windows产生一个窗口时(还有许多其它类似,请参考<<深入浅出MFC>>第9章,563页),就会调用你设定的Hook函数。
  这样设定完成后,回到CWnd::CreateEx函数中,执行::CreateWindowEx进行窗口创建,窗口一产生,就会调用上面设定的Hook函数_AfxCbtFilterHook。而正是在_AfxCbtFilterHook中对函数PreSubclassWindow进行了第二次调用。见如下代码:
none.gif // FromVSInstallPathVC98MFCSRCWINCORE.CPP
expandedblockstart.gifcontractedblock.gif
/**/ /**/ /**/ /////
none.gif // Windowcreationhooks
none.gif

none.gifLRESULTCALLBACK
none.gif_AfxCbtFilterHook(
int code,WPARAMwParam,LPARAMlParam)
expandedblockstart.gifcontractedblock.gifdot.gif
dot.gif {
inblock.gifdot.gif
inblock.gifdot.gif
inblock.gif
//connecttheHWNDtopWndInitdot.gif
inblock.gif
pWndInit->Attach(hWnd);
inblock.gif
//allowothersubclassingtooccurfirst
inblock.gif
pWndInit->PreSubclassWindow();
inblock.gifdot.gif
expandedsubblockstart.gifcontractedsubblock.gifdot.gif
dot.gif{
inblock.gif
//subclassthewindowwithstandardAfxWndProc
inblock.gif
oldWndProc=(WNDPROC)SetWindowLong(hWnd,GWL_WNDPROC,(DWORD)afxWndProc);
inblock.gifASSERT(oldWndProc
!=NULL);
inblock.gif
*pOldWndProc=oldWndProc;
expandedsubblockend.gif}

inblock.gifdot.gif
expandedblockend.gif}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值