ATL和MFC的C++类和HWND的映射机制

最近看深入解析ATL这本书的时候看到ATL中的窗口类实现的时候,很是惊异于ATL的Thunk的运用技术,,ATL运用Thunk技术把C++的成员函数置换成Windows窗口的消息处理函数。那么更古老的MFC框架又是怎么实现CWnd类到HWND窗口类之间的映射的呢? 下面的文章将ATL和MFC的窗口封装机制做个对比.


如果让我完成C++类到HWND窗口的映射,我会写如下的代码

BOOL CMyWnd::Create(...)

{

...

HWND hWnd = ::CreateWindowEx(...);

this->m_hWnd = hWnd;

//置换hWnd窗口过程

...

}

我相信很多程序员都会像这样封装的,但是这样有一个很大的问题,API函数CreateWindowEx在创建返回之前,Windows已经发送了一些消息给创建好的窗口,其中包括很重要的WM_CREATE消息,而将C++的成员函数置换hWnd的窗口例程是在CreateWindowEx返回之后,因此C++的成员函数就没办法处理WM_CREATE消息,这可就不行了。看看MFC和ATL是怎么完成的。


1. MFC的实现过程:

MFC的CWnd封装了Windows窗口,其创建函数为CreateEx成员函数,CreateEx函数在调用CreateWindowEx方法之前首先通过AfxHookWindowCreate函数建立一个CBT钩子,该钩子的作用在于钩取创建窗口的事件.赶在创建窗口完成之前建立HWND到CWnd的全局映射,之后置换窗口处理函数为AfxWndProc标准函数,这样做的目的在于CreateEx函数内部调用CreateWindowEx API创建HWND, CreateWindowEx函数返回之后调用AfxUnhookWindowCreate卸载CBT钩子。

AfxWndProc函数过程:

从全局映射中取出与句柄HWND相对应的CWnd对象,找到后调用CWnd:: WindowProc的成员函数,剩下的调用就在于CWnd中消息和消息处理函数的映射表了.


2.ATL的实现过程:

ATL中的宏DECLARE_WND_CLASS为CWindowImpl类提供Windows窗口的注册信息,其定义如下:

#define DECLARE_WND_CLASS(WndClassName) "

static ATL::CWndClassInfo& GetWndClassInfo() "

{ "

static ATL::CWndClassInfo wc = "

{ "

{ sizeof(WNDCLASSEX), CS_HREDRAW | CS_VREDRAW | CS_DBLCLKS, StartWindowProc, "

0, 0, NULL, NULL, NULL, (HBRUSH)(COLOR_WINDOW + 1), NULL, WndClassName, NULL }, "

NULL, NULL, IDC_ARROW, TRUE, 0, _T("") "

}; "

return wc; "

}

通过宏代码可以看出,其提供给注册的窗口例程为StartWindowProc函数, StartWindowProc函数代码如下:

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值