http://blog.csdn.net/swimmer2000/archive/2007/10/30/1856213.aspx MFC(VC6.0)的CWnd及其子类中,有如下三个函数:
classCWnd:publicCCmdTarget
{
public:
virtualBOOLPreCreateWindow(CREATESTRUCT&cs);
virtualvoidPreSubclassWindow();
BOOLSubclassWindow(HWNDhWnd);
};
让人很不容易区分,不知道它们究竟干了些什么,在什么情况下要改写哪个函数?
想知道改写函数?让我先告诉你哪个不能改写,那就是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中的这三个函数都是这样实现的吧!
BOOLCWnd::CreateEx(DWORDdwExStyle,LPCTSTRlpszClassName,
LPCTSTRlpszWindowName,DWORDdwStyle,
intx,inty,intnWidth,intnHeight,
HWNDhWndParent,HMENUnIDorHMenu,LPVOIDlpParam)
{
//allowmodificationofseveralcommoncreateparameters
CREATESTRUCTcs;
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();
returnFALSE;
}
AfxHookWindowCreate(this);
HWNDhWnd=::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);
returnTRUE;
}
//forchildwindows
BOOLCWnd::PreCreateWindow(CREATESTRUCT&cs)
{
if(cs.lpszClass==NULL)
{
//makesurethedefaultwindowclassisregistered
VERIFY(AfxDeferRegisterClass(AFX_WND_REG));
//noWNDCLASSprovided-usechildwindowdefault
ASSERT(cs.style&WS_CHILD);
cs.lpszClass=_afxWnd;
}
returnTRUE;
}
CWnd::CreateEx先设定cs(CREATESTRUCT),在调用真正的窗口创建函数::CreateWindowEx之前,调用了CWnd::PreCreateWindow函数,并把参数cs以引用的方式传递了进去。而CWnd的PreCreateWindow函数也只是给cs.lpszClass赋值而已。毕竟,窗口创建函数CWnd::CreateEx的诸多参数中,并没有哪个指定了所要创建窗口的窗口类,而这又是不可缺少的(请参考<<windows程序设计>>第三章)。所以当你需要修改窗口的大小、风格、窗口所属的窗口类等cs成员变量时,要改写PreCreateWindow函数。
//FromVSInstallPathVC98MFCSRCWINCORE.CPP
BOOLCWnd::SubclassWindow(HWNDhWnd)
{
if(!Attach(hWnd))
returnFALSE;
//allowanyothersubclassingtooccur
PreSubclassWindow();
//nowhookintotheAFXWndProc
WNDPROC*lplpfn=GetSuperWndProcAddr();
WNDPROColdWndProc=(WNDPROC)::SetWindowLong(hWnd,GWL_WNDPROC,
(DWORD)AfxGetAfxWndProc());
ASSERT(oldWndProc!=(WNDPROC)AfxGetAfxWndProc());
if(*lplpfn==NULL)
*lplpfn=oldWndProc;//thefirstcontrolofthattypecreated
#ifdef_DEBUG
elseif(*lplpfn!=oldWndProc)
{
::SetWindowLong(hWnd,GWL_WNDPROC,(DWORD)oldWndProc);
}
#endif
returnTRUE;
}
voidCWnd::PreSubclassWindow()
{
//nodefaultprocessing
}
CWnd::SubclassWindow先调用函数Attach(hWnd)让CWnd对象和hWnd所指的窗口发生关联。接着在用::SetWindowLong修改窗口过程(subclass)前,调用了PreSubclassWindow。CWnd::PreSubclassWindow则是什么都没有做。
在CWnd的实现中,除了CWnd::SubclassWindow会调用PreSubclassWindow外,还有一处。上面所列函数CreateEx的代码,其中调用了一个AfxHookWindowCreate函数,见下面代码:
//FromVSInstallPathVC98MFCSRCWINCORE.CPP
BOOLCWnd::CreateEx()
{
//allowmodificationofseveralcommoncreateparameters
if(!PreCreateWindow(cs))
{
PostNcDestroy();
returnFALSE;
}
AfxHookWindowCreate(this);
HWNDhWnd=::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);
returnTRUE;
}
接着察看AfxHookWindowCreate的代码:
//FromVSInstallPathVC98MFCSRCWINCORE.CPP
voidAFXAPIAfxHookWindowCreate(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进行了第二次调用。见如下代码:
//FromVSInstallPathVC98MFCSRCWINCORE.CPP
/**//**//**////////////////////////////////////////////////////////////////////////////////Windowcreationhooks
LRESULTCALLBACK
_AfxCbtFilterHook(intcode,WPARAMwParam,LPARAMlParam)
{
//connecttheHWNDtopWndInit
pWndInit->Attach(hWnd);
//allowothersubclassingtooccurfirst
pWndInit->PreSubclassWindow();
{
//subclassthewindowwithstandardAfxWndProc
oldWndProc=(WNDPROC)SetWindowLong(hWnd,GWL_WNDPROC,(DWORD)afxWndProc);
ASSERT(oldWndProc!=NULL);
*pOldWndProc=oldWndProc;
}
}
相关推荐
怎样将CWnd转换为HWND和HWND转换为CWnd
CWnd类提供了微软基础类库中所有窗口类的基本功能。 CWnd对象与Windows的窗口不同,但是两者有紧密联系。CWnd对象是由CWnd的构造函数和析构函数创建或销毁的。另一方面,Windows的窗口是Windows的一种内部数据结构,...
学习VC++必备,CWnd类所有函数的中文翻译,共300多个函数,超赞!
在VC++ MFC中 如何通过HWND获得CWnd指针
学习VC++必备,CWnd类所有函数的中文翻译,共进30个函数,另还有CDocument类、CDC类、CCmDUI类和CDiglog类的翻译资源,需要的可另行搜索,超赞!
自己整理的CWnd类成员函数详解,摘自《MFC类库详解》。
从CWnd直接派生的按钮控件
在MFC的窗口中,如何通过HWND获得CWnd指针
从cwnd派生,自绘的一个QQ好友分组控件,程序运行起来后,需要自己现添加分组,然后在分组中添加成员。
根据crkchina在http://topic.csdn.net/u/20081023/22/99b0d49a-d07b-4837-a8df-7f9c4b629bd3.html中对子窗体的要求,写的一个基于CWnd类的透明窗体。 效果还是不进满意,只能等待高手给出更好的意见了。 代码都是...
CWnd-MFC中文手册.pdf
要想通过CWnd类控制窗口,必须先取得CWnd类(或派生类)的对像或指针。主要方法有四: 一,系统已经定义好的类。 二,自己定义对像,并操作它。 三,利用类向导为控件关联变量(前面已讲,不再详细讲解)。 四,利用...
21个比较重要的Cwnd类成员例子,21个比较重要的Cwnd类成员例子
各种CWND类成员函数列表及相关功能描述
OnVScroll的自定义实现,cwnd派生类如何create
基于MFC的控件自绘,从CWnd继承,当前工程是自绘成箭头形状