摘 要:本文介绍了窗口子类化(SubClassing)的概念、技术原理、作用以及在Visual C++6.0中的实现方法,并给出了一个具体应用实例。实践证明,适当地使用窗口子类化技术,可以大大增强应用程序的功能。
关键字:子类化, 窗口函数, 消息, Visual C++
一、引言
在Windows编程中,如果我们想在窗口程序执行时改变它所包含的控件(对话框中的按钮、下拉式菜单等)的某些行为,采用窗口子类化技术是一个不错的选择。可以使用对已有控件派生子类的方式定义一个子类,而控件的消息处理则在新定义的子类里定义。适当使用子类化技术创建出容易使用的新窗口类,往往可以使你的程序界面更具人性化。
二、窗口子类化技术概述[1]
Windows的窗口类是一个窗口模板,包含一个窗口所具有的部分窗口属性。编写一个Windows程序时首先要做的工作就是注册一个窗口类,然后基于此注册的窗口类创建一个新的窗口。在WIN32平台中,注册窗口类的API函数是RegisterClass和RegisterClassEX,其中RegisterClassEX是推荐使用的函数,使用这个函数注册窗口类时,需要先填写一个WNDCLASSEX结构。这个结构实际上反映了一个窗口类的特征,一个窗口类有本类所有窗口公用的类属性、窗口函数、类图标和小图标、类鼠标、窗口背景刷、类菜单,当然还有类名。除此之外,每个类还有一定大小的类存储区,可以用来存储该类的公共数据。
每一个创建的窗口都有一个窗口函数,其地址由WNDCLASSEX结构的lpfnWndProc参数设定,该窗口函数处理对应于该窗口类的所有实例的消息。当创建一个窗口时,Windows将分配一个内存块,用来存放与该窗口相关的信息,并将参数lpfnWndProc从窗口类内存块拷贝到该内存块中。当消息被分发到窗口时,Windows检查该窗口中内存块中的lpfnWndProc值,并调用该内存块地址上的窗口函数。
一个窗口的行为主要取决于它的窗口函数,如果能够改变一个窗口的窗口函数,使它指向自己写的某个函数,那就意味着发给这个窗口的各种消息将由我们自己写的这个函数来处理。
子类化一个窗口,实际上就是改变窗口内存块中的窗口函数的地址,使其指向用户自定义的新的窗口函数入口,以实现用户希望的窗口特性。
三、窗口子类化的作用
窗口子类化技术最大的特点就是能够截取Windows的消息。一旦用户自定义的窗口函数截取了传向原窗口函数的消息,就可以对被截取的消息进行如下处理[2]:
n 将其传给原来的窗口函数。这是对大多数消息应该采取的措施,因为子类通常只对原来的窗口特性作少量的修改。
n 截取该消息,阻止其向原窗口函数发送。
n 修改该消息,修改完毕以后再向原窗口函数发送。
Windows SDK提供了一些设计好的窗口类,如EDIT、LISTBOX、TREEVIEW等。通过截取这些通用窗口类的消息,用户程序可以为它们添加新的特性,改善其外观,扩充其功能。
子类化的优点主要体现在以下两个方面:首先,它不需要创建新的窗口类,不需要了解一个窗口的窗口过程。这在原来的窗口函数是由别人编写,而且创建过程不可见的情况下非常有用;其次,子类化比较容易实现,因为所有要做的工作仅仅就是写一个窗口函数。
四、在VC中实现窗口子类化
上面介绍的子类化是从Windows本身的窗口函数概念来讲的,实际上属于SDK(Software Development Kit)编程的范畴,在MFC中情况有所不同。下面将分别描述在这两种情况下窗口子类化实现的方法。
4.1 VC中基于SDK编程的窗口子类化
VC中基于SDK编程的窗口子类化的基本步骤如下:
(1) 正常创建原始窗口,得到窗口的句柄。
(2) 调用GetWindowLong得到原来的窗口函数OldWndProc。
(3) 调用SetWindowLong设置新的窗口函数NewWndProc。
新的窗口函数的代码如下所示:
LRESULT NewWndProc(HWND hWnd,UINT message,WPARAM wParam,LPARAM lParam)
{
if(message==WM_IcareIt)
{
//截取自己感兴趣的消息,作一些处理,达到改变特性的目的
}
//必要时可以调用原来的窗口函数,使被子类化的窗口仍具有原来的很多特性
return CallWndowProc(OldWndProc,hWnd,message,wParam,lParam);
}
值得注意的是,在调用旧的窗口函数时,不能直接用OldWndProc(…),而必须用函数CallWndProc进行调用,否则会出现堆栈错误。
4.2 MFC编程中的窗口子类化
MFC窗口实际上已经是被子类化的窗口。所有的MFC窗口共享同一个窗口函数,由这个窗口函数根据窗口句柄,查找这个窗口对应的CWnd派生类实例,再通过消息映射这个窗口类的消息处理函数。鉴于以上原因,在MFC中要子类化一个窗口就比较容易了,因为你的任务只是编写一个新的MFC窗口类而不需要写一个窗口函数。
假如我们现在有一个对话框,里面有一个编辑控件,我们只希望在该控件中接受非数字字符输入,我们可以拦截WM_CHAR消息,在它的处理函数中忽略任何数字的输入。MFC编程中窗口子类化的具体实现步骤在下一节笔者将用一个简单的实例来加以说明。
五、VC中窗口子类化的应用举例
MFC为广大编程者提供了很多功能丰富的窗口类,如果能在这些通用窗口类的基础上进行子类化的话,将会给编程者带来很多便利。下面举一个例子来说明MFC编程中的子类化是多么的简单易行。该例完成上面提到的在编辑控件只接受非数字字符输入的功能。实现这个子类化的基本步骤和相关代码如下:
(1)利用AppWziard创建一个基于对话框的程序SubClassing。
(2)对MFC提供的标准的对话框中的控件进行修改,删除MFC提供的静态文本控件,添加自己的一个编辑控件,设置新控件的ID为IDC_EDIT。合理布置对话框上各控件的位置,使程序界面布局合理、美观。
(3)用ClassWizard从CEdit类派生一个新的窗口类,新窗口的窗口类叫CNoNumEdit。截取CNoNumEdit类的WM_CHAR消息,在OnChar函中完成忽略任何数字的输入的处理。实现代码如下:
void CNoNumEdit::OnChar(UINT nChar, UINT nRepCnt, UINT nFlags)
{
TCHAR ch=nChar;
if(ch>=_T('0')&&ch<=_T('9'))
{
AfxMessageBox(("请不要输入数字!"),MB_OK);
//当输入数字字符时将被忽略,并显示警告信息
return;
}
CEdit::OnChar(nChar, nRepCnt, nFlags);//输入为非数字字符时调用原处理函数
}
(4)在对话框窗口类CSubClassingDlg的定义中添加变量CNoNumEdit ed。在CSubClassingDlg::OnInitDialog()函数中调用CWnd类的成员函数SubClassWindow进行子类化。
ed.SubclassWindow(GetDlgItem(IDC_EDIT)->m_hWnd);
(5) 在对话框窗口类CsubClassing的OnDestroy中调用ed.UnSubClassWindow()执行窗口类的反子类化。
现在可以编译执行这个程序了,当用户输入数字字符时将会忽略该输入,并显示警告信息。
六、结束语
在Windows编程中,适当使用窗口子类化技术,可以很方便地达到改变一个窗口的特性的目的。当然子类化也存在其局限性。实际上,子类化的概念是针对一个已经创建的窗口来谈的,所以修改窗口函数是在窗口创建之后进行的,在窗口创建期间的消息无法捕获,也就无法处理。另外有些窗口的特性与窗口类本身的属性有关。比如如果一个窗口类没有CS_DBLCLKS属性的话,那么要想通过子类化这些窗口达到处理WM_LBUTTONDBLCLK消息的目的是无法实现的。对于子类化的以上局限性,可以通过超类化(SuperClassing)技术消除。有兴趣的读者可以参阅文献[1]。
参考文献
[1] 陈俊,郑静. Visual C++中窗口子类化和超类化技术的应用. 现代计算机[J],2002(5):79~82
[2] 邓双成,田海晏.VB中窗口子类化技术的实现和应用.计算机应用[J],2000,20(12):53~55
|