C++ 中超类化和子类化
超类化和子类化没有具体的代码,其实是一种编程技巧,在MFC和WTL中可以有不同的实现方法。
窗口子类化:
原理就是改变一个已创建窗口类的窗口过程函数。通过截获已创建窗口的消息,从而实现监视或修改已创建窗口类的行为属性。可以用来改变或者扩展一个已存在的窗口的行为,而不用重新开发。比如要获得那些预定义控件窗口类(按钮控件、编辑控件、列表控件、下 拉列表控件、静态控件和滚动条控件)的功能而又要修改它们的某些行为。
子类化的优点主要体现在以下两个方面:首先,它不需要创建新的窗口类,不需要了解一个窗口的窗口过程。这在原来的窗口函数是由别人编写,而且创建过程不可见的情况下非常有用;其次,子类化比较容易实现,因为所有要做的工作仅仅就是写一个窗口函数。
主要步骤为
截取该消息,阻止其向原窗口函数发送。
修改该消息。
修改完毕以后再向原窗口函数发送。
// 保存窗口默认的消息响应函数指针 WNDPROC pSubclassOldEditProc; // 用于替换子类化窗口的消息响应函数 LRESULT CALLBACK JcEditProcSubClass(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) { switch(message) { case WM_CHAR: { ::MessageBox(hWnd, "WM_CHAR响应", "子类化", MB_OK); return 0; } //使用完后,消息发回原窗体 default: return ::CallWindowProc(pSubclassOldEditProc, hWnd, message, wParam, lParam); } } // 对创建好的窗体进行子类化代码 { // 创建 HWND hEdit = CreateWindowEx(NULL, "EDIT", "SubClass", WS_CHILD|WS_BORDER|ES_LEFT|ES_AUTOHSCROLL, 100,120, 128, 16, hWnd, NULL, hInstance, NULL); //修改窗口属性,改变消息响应函数 pSubclassOldEditProc = (WNDPROC)::SetWindowLong(hEdit, GWL_WNDPROC, (DWORD)JcEditProcSubClass); // 显示 ShowWindow(hEdit, nCmdShow); UpdateWindow(hWnd); }
窗口超类化:
窗口超类化是在窗口类――WNDCLASS或WNDCLASSEX(非MFC类概念)级别进行的改变窗口类特征的。改变已有窗口类的行为属性。
通过调用 GetClassInfoEx 来获得想要进行超类化操作的窗口类的信息。函数GetClassInfoEx 需要一个指向 WNDCLASSEX 结构的指针,用于当成功返回时填入窗口类的信息。
按需要修改 WNDCLASSEX 结构的成员,其中有两个成员必须修改:
hInstance 存放程序的实例句柄
lpszClassName 指向一个新类名的指针
不必修改成员 lpfnWndProc,但大多数情况下还是需要的。但要记住如果要使用函数 CallWindowProc 调用老窗口的过程,那就必须保存成员 lpfnWndProc 的原值。
注册修改完的 WNDCLASSEX 结构,得到一个具有旧窗口类某些特性的新窗口类。
用新窗口类创建窗。
WNDPROC pSuperOldEditProc;// 保存窗口默认消息处理函数 // 用于替换的超类化消息响应函数 LRESULT CALLBACK JcEditProcSuper(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) { switch(message) { case WM_CHAR: { ::MessageBox(hWnd, "WM_CHAR响应", "超类化", MB_OK); return 0; } default: return ::CallWindowProc(pSuperOldEditProc, hWnd, message, wParam, lParam); } } // 创建超类化控件代码 { // 取得原控件信息 WNDCLASSEX myeditClass; ::GetClassInfoEx(hInstance, "EDIT", &myeditClass); // 保存原控件默认消息处理函数 pSuperOldEditProc = myeditClass.lpfnWndProc; // 设置替换的消息处理函数 myeditClass.lpfnWndProc = JcEditProcSuper; // 指定新的窗口类名字 myeditClass.lpszClassName = "JcilyEdit"; // 设置结构体大小 myeditClass.cbSize = sizeof(WNDCLASSEX); // 注册新信息 RegisterClassEx(&myeditClass); // 创建 HWND hEdit = CreateWindowEx(NULL, myeditClass.lpszClassName, "SuperClass", WS_CHILD|WS_BORDER|ES_LEFT|ES_AUTOHSCROLL, 100,100, 128, 16, hWnd, NULL, hInstance, NULL); // 显示 ShowWindow(hEdit, nCmdShow); UpdateWindow(hWnd); }
窗口子类化和超类化的区别
(1) 子类化修改窗口过程函数, 超类化修改窗口类(新的窗口类名)
(2) 子类化是在窗口实例级别上的,超类化是在窗口类(WNDCLASS)级别上的。
(3) 超类化可以完成比子类化更复杂的功能,在SDK范畴上,可以认为子类化是超类化的子集。
(4) 子类化只能改变窗口创建后的性质,对于窗口创建期间无能为力(无法截获ON_CREATE 事件),而超类化可以实现;超类化不能用于Windows创建的窗口,子类化可以。
(5) 超类化可以修改包含窗体背景等属性,而子类化不能。
C++ 超类化和子类化的区别
Q: 我看了WINX开发包中的文档中,你提到了超类,它是什么概念?我平常只听过子类(Subclass)。
A: 子类化(Subclass)是指替换窗口过程(WNDPROC)。
超类(Superclass)是替换窗口过程(WNDPROC),并且替换ClassName。
Subclass不太象继承,而像是一种外挂(Hook)行为。
Superclass则更像继承,因为生成了新的窗口类,并且继承了行为。
winx中Subclass和Superclass用同一个类实现。都是winx::SubclassWindow。
其他界面库一般不提供Superclass。
Q: 你说其他界面库不提供Superclass?MFC里继承应该属于Superclass吧?
A: 不,MFC中用的是Subclass。
Q: 我怎么感觉不出二者的区别?我是说用法上和结果上。
A: 两者在用法上有异,但获得的结果确实无太大差别。
我们以Button为例。如果是Subclass,那么用户先要有一个Button,然后Subclass它。
也就是说Subclass发生在CreateWindow之后。
如果是Superclass,那么用户CreateWindow的时候直接传入新的窗口类名称,根本就没有Button被生成。
当然,这要求CreateWindow之前调用过该窗口类的RegisterClass。
MFC用的是Subclass。因为CButton类通过DDX技术和对话框上的Button关联的。
也就是说,MFC中是先有了Button,然后由DDX技术Subclass它。
所以,一般你看不到Subclass流程。
Q: 什么情况下需要用SuperClass?
哦,我明白了,是不是有了SuperClass技术,用户可以很方便的创建自己的Control?
A: 对了。不需要从零开始。
Q: 对。感觉多数用户自定义控件还是和系统控件有关系的。
A: 是的。这正是Superclass存在的意义。
你可以想象一下:在以前,你提供一个控件,你要告诉它,先创建Button,然后调用我的Subclass函数。
而有了Superclass,现在你只需要告诉它窗口类的名字就可以了。
因为Superclass隐蔽了你从Button继承这个事实。
Q: 对。
A: 之所以winx可以有superclass而其他界面库没有,究其原因,还是与屏蔽“窗口类”这个出发点有关。
Q: 对,对,对。以后自己定义的控件也可以可视化开发。
A: 是的。
Q: 指定一下我们自己的窗口类就可以了。
WINX这种控件感觉是介于系统控件和ActiveX控件之间的一种控件。
A: 是的。这种控件其实就是Windows系统控件的实现方式。
只是系统控件不需要你主动注册,Windows已经帮你注册好了。
Q: 系统实现这些系统控件不是通过SuperClass吧?它应该是从最一般化的窗口继承而来。
A: 呵呵,当然不是。它们没有什么可以借用的,只好从最基础的winx::Window继承。
Q: 不管是白手起家,还是有点基础,做法都是SuperClass。
因为winx::Window的基础是DefWindowProc,也是一个窗口过程。
A: 可以这么理解。
Q: 是不是可以这样理解,SubClass只是更改了WndProc,而SuperClass还更改了其他窗口属性?
A: 是的。你的理解完全正确,这正是Subclass与Superclass最本质的区别。
Superclass可以改窗口类(WNDCLASSEX)的任何数据。WINX就是这么实现的。
子类化样例代码:
// ------------------------------------------------------------------------- // class CMyEdit - 使用子类化(Subclass)技术 class CMyEdit : public winx::Edit{ public: VOID OnContextMenu(HWND hWnd, winx::CPoint pt) { //禁止了右键菜单... } }; // ------------------------------------------------------------------------- // CHelloDlg class CHelloDlg : public winx::ModalDialog { public: BOOL OnInitDialog(HWND hDlg, HWND hWndDefaultFocus) { CMyEdit::DoSubclassDlgItem(hDlg, IDC_EDIT1); return TRUE; } }; // ------------------------------------------------------------------------- int APIENTRY WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow) { CHelloDlg dlg; dlg.DoModal(); return 0; } // -------------------------------------------------------------------------
超类化样例代码:
// ------------------------------------------------------------------------- // class CMyEdit2 - 使用超类化(Superclass)技术 class CMyEdit2 : public winx::Edit{ WINX_CLASS("MyEdit"); public: VOID OnContextMenu(HWND hWnd, winx::CPoint pt) { //禁止了右键菜单... } }; // ------------------------------------------------------------------------- // CHelloDlg class CHelloDlg : public winx::ModalDialog ; // ------------------------------------------------------------------------- int APIENTRY WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow) { CMyEdit2::RegisterClass(); CHelloDlg dlg; dlg.DoModal(); return 0; }
永劫无间手游测试服 安卓版v1.0.262342
下载永劫无间手游台服 安卓版v1.0.262342
下载永劫无间手游国际服 安卓版v1.0.262342
下载永劫无间手游豌豆荚版 安卓版v1.0.262342
下载曼德拉男孩 最新版v2025.1.1
曼德拉男孩是一款专为女性玩家准备的治愈类恋爱游戏,在这里玩家
游戏开发者无限金币版 最新版v1.0.16
游戏开发者内置菜单版是一款非常好玩的模拟经营类手游,内部有功
没有中间商赚差价内购版 最新版v23.7.3
没有中间商赚差价免广告是一款非常好玩的模拟经营类手游,无需看
我的世界某不科学的空岛下载mcbbs 最新版v隔壁老王
我的世界某不科学的空岛整合包是一款像素风格的模拟沙盒游戏,该
洗衣店模拟器无限钞票免广告版 v2.2.2
洗衣店模拟器无限钞票版是一款模拟经营类手游,玩家们将在游戏中