【转】Windows消息投递流程:一般窗口消息投递(WM_LBUTTONCLICK)

原文网址:http://blog.csdn.net/hyhnoproblem/article/details/6182646

本例通过在单文档程序的视图中添加WM_LBUTTONCLICK消息处理函数,来解释一般窗口消息的投递流程。 基于VS 2005

[cpp] view plaincopy

  1. BEGIN_MESSAGE_MAP(CMyView, CView)
  2. ON_WM_LBUTTONDBLCLK()
  3. END_MESSAGE_MAP()
  4. // ON_WM_LBUTTONDBLCLK宏展开
  5. #define ON_WM_LBUTTONDBLCLK() /
  6. { WM_LBUTTONDBLCLK, 0, 0, 0, AfxSig_vwp, /
  7. (AFX_PMSG)(AFX_PMSGW) /
  8. (static_cast< void (AFX_MSG_CALL CWnd::*)(UINT, CPoint) > ( &ThisClass :: OnLButtonDblClk)) },

从上面的代码可以看出WM_LBUTTONCLICK消息的类型标签是AfxSig_vwp, 在afxmsg_.h中有一个枚举类型AfxSig,消息标签主要用于区分消息处理函数的类型。该类型中定义了AfxSig_vwp的值:

[cpp] view plaincopy

  1. enum AfxSig
  2. {
  3. //...
  4. AfxSig_vWp = AfxSig_v_W_p
  5. //...
  6. }

在AfxWndProc中,将消息中的句柄映射成窗口类指针,这个指针指向CMyView。AfxWndProc调用AfxCallWndProc,AfxCallWndProc调用CWnd::WindowProc,CWnd::WindowProc调用CWnd::OnWndMsg,CWnd::OnWndMsg完成对消息的处理。

[cpp] view plaincopy

  1. // wincore.cpp 1746
  2. BOOL CWnd::OnWndMsg(UINT message, WPARAM wParam, LPARAM lParam, LRESULT* pResult)
  3. {
  4. // ...
  5. // GetMessageMap是个虚函数,因为当前指针是指向CMyView,所以取到的是CMyView的消息映射表
  6. const AFX_MSGMAP* pMessageMap; pMessageMap = GetMessageMap();
  7. UINT iHash; iHash = (LOWORD((DWORD_PTR)pMessageMap) ^ message) & (iHashMax-1);
  8. winMsgLock.Lock(CRIT_WINMSGCACHE);
  9. // 全局消息散列缓存,查找缓存
  10. AFX_MSG_CACHE* pMsgCache; pMsgCache = &_afxMsgCache[iHash];
  11. const AFX_MSGMAP_ENTRY* lpEntry;
  12. if (message == pMsgCache->nMsg && pMessageMap == pMsgCache->pMessageMap)
  13. {
  14. // cache hit
  15. lpEntry = pMsgCache->lpEntry;
  16. winMsgLock.Unlock();
  17. if (lpEntry == NULL)
  18. return FALSE;
  19. // cache hit, and it needs to be handled
  20. if (message < 0xC000)
  21. goto LDispatch;     // 系统消息?
  22. else
  23. goto LDispatchRegistered;       // 已注册消息?
  24. }
  25. else
  26. {
  27. // not in cache, look for it
  28. pMsgCache->nMsg = message;
  29. pMsgCache->pMessageMap = pMessageMap;
  30. for (/* pMessageMap already init‘ed */; pMessageMap->pfnGetBaseMap != NULL;
  31. pMessageMap = (*pMessageMap->pfnGetBaseMap)())
  32. {
  33. // Note: catch not so common but fatal mistake!!
  34. //      BEGIN_MESSAGE_MAP(CMyWnd, CMyWnd)
  35. ASSERT(pMessageMap != (*pMessageMap->pfnGetBaseMap)());
  36. if (message < 0xC000)
  37. {
  38. // constant window message
  39. if ((lpEntry = AfxFindMessageEntry(pMessageMap->lpEntries,
  40. message, 0, 0)) != NULL)
  41. {
  42. pMsgCache->lpEntry = lpEntry;
  43. winMsgLock.Unlock();
  44. goto LDispatch;
  45. }
  46. }
  47. else
  48. {
  49. // registered windows message
  50. lpEntry = pMessageMap->lpEntries;
  51. while ((lpEntry = AfxFindMessageEntry(lpEntry, 0xC000, 0, 0)) != NULL)
  52. {
  53. UINT* pnID = (UINT*)(lpEntry->nSig);
  54. ASSERT(*pnID >= 0xC000 || *pnID == 0);
  55. // must be successfully registered
  56. if (*pnID == message)
  57. {
  58. pMsgCache->lpEntry = lpEntry;
  59. winMsgLock.Unlock();
  60. goto LDispatchRegistered;
  61. }
  62. lpEntry++;      // keep looking past this one
  63. }
  64. }
  65. }
  66. pMsgCache->lpEntry = NULL;
  67. winMsgLock.Unlock();
  68. return FALSE;
  69. }
  70. // ...
  71. LDispatch:
  72. mmf.pfn = lpEntry->pfn;
  73. switch (lpEntry->nSig)
  74. {
  75. //...
  76. case AfxSig_v_u_p:      // 消息标签
  77. {
  78. CPoint point(lParam);
  79. (this->*mmf.pfn_v_u_p)(static_cast<UINT>(wParam), point);
  80. }
  81. break;
  82. //...
  83. }
  84. //...
  85. LDispatchRegistered:    // for registered windows messages
  86. ASSERT(message >= 0xC000);
  87. ASSERT(sizeof(mmf) == sizeof(mmf.pfn));
  88. mmf.pfn = lpEntry->pfn;
  89. lResult = (this->*mmf.pfn_l_w_l)(wParam, lParam);
  90. }
  91. // 消息处理函数类型枚举
  92. union MessageMapFunctions
  93. {
  94. AFX_PMSG pfn;   // generic member function pointer
  95. // ...
  96. LRESULT (AFX_MSG_CALL CWnd::*pfn_l_w_l)(WPARAM, LPARAM);
  97. };
时间: 2024-10-05 12:20:26

【转】Windows消息投递流程:一般窗口消息投递(WM_LBUTTONCLICK)的相关文章

【转】Windows消息投递流程:WM_COMMAND消息流程

原文网址:http://blog.csdn.net/hyhnoproblem/article/details/6182585 该示例通过研究基本的单文档程序的“文件”--“打开”命令,分析WM_COMMAND消息投递流程.基于VS 2005 代码 AfxWndProc最终调用的是OnWndMsg,这个函数负责消息的分发处理.当消息是WM_COMMAND时,将消息投递给OnCommand函数. [cpp] view plaincopy // wincore.cpp 1746 BOOL CWnd::

Windows窗口消息大全

Windows窗口消息大全,全不全自己看 ////////////////////////////////////////////////////////////////////////// #include "AFXPRIV.H"//消息值的定义来源 #include "Dde.h"//DDE消息值的定义来源 #include "CPL.H"//控制面板消息值的定义来源 #include "WFEXT.H"//文件管理器的消

消息循环,注册窗口,创建窗口【图解】

本文是Win32编程基础 尽管Windows应用程序千变万化,令人眼花缭乱,但,消息机制和窗口过程却始终它们的基础,掌握了这两项技术,也就相当于把握住了问题的关键 DirectX编程也是建立在这个基础之上的,所以,在你可以熟练的进行简单的Win32编程之后DirectX编程也就触手可得. 1.1 简单的WIN32程序 在以前的C语言编程中,一个最简单的程序可以只有两行. void main(void) { printf "Hello World!"; } 而要实现同样功能的Window

Duilib中的消息泵和虚拟窗口

Duilib中的消息泵和虚拟窗口 一.消息泵的结构 CNotifyPump类是构建Duilib消息泵的根父类,要使用消息泵机制的窗口类应该从该类继承.在继承关系的基础上,通过DUI_DECLARE_MESSAGE_MAP.DUI_BEGIN_MESSAGE_MAP.DUI_END_MESSAGE_MAP.DUI_ON_MSGTYPE.DUI_ON_MSGTYPE_CTRNAME.DUI_ON_CLICK_CTRNAME.DUI_ON_SELECTCHANGED_CTRNAME.DUI_ON_KI

MFC窗口消息PostMessage和SendMessage

以前这些消息用得比较少,但是今天碰到了个事儿,我看非用消息不可. 事情是这样的,我在线程中需要刷新对话框上面的内容,但是每每执行到UpdateData时就出现了断言错误. 查了相关资料,发现这个可能是多个模块调用同一个方法出现的问题.反正照我之前的逻辑走不通. 我现在将更新对话框信息的函数,通过线程中发窗口消息,让对话框的消息处理函数来负责处理我自定义的消息,为我刷新对话框数据. // http://blog.csdn.net/a8082649/article/details/7733527 1

Win32窗口消息机制 x Android消息机制 x 异步执行

如果你开发过Win32窗口程序,那么当你看到android代码到处都有的mHandler.sendEmptyMessage和 private final Handler mHandler = new Handler() { @Override public void handleMessage(Message msg) { switch (msg.what) { case MSG_REPORT_PRIMARY_CLIP_CHANGED: reportPrimaryClipChanged(); }

spy++捕获窗口消息

打开spy++,窗口截图如下,点击窗口搜索按钮(红框标识) ,如果找不到对应的窗口,鼠标右键刷新即可. 鼠标左键点击窗口搜索图标,按住不放,拖到需要抓取消息的窗口上: spy++会自动在列表中高亮定位抓取到的窗口,如下图所示,notepad++的new1窗口被抓取到了.再点击菜单栏 监视->日志消息: 在弹窗的消息选项弹窗中,可以选择需要打印的日志的消息,可以通过消息组选择一类消息,也可以在左侧"要查看的消息"中,通过鼠标左键点击选择和取消选择,也可以通过连续按键盘字母键查找对应

8.windows消息机制(三)消息队列

1.消息队列 消息队列用于存放消息的一个队列,消息在队列中先入先出.所有窗口程序都具有消息队列,程序可以从队列中获取消息. 2.消息队列的类型 系统消息队列 - 由系统维护的消息队列,存放系统产生的消息,例如鼠标.键盘等. 程序消息队列 - 属于每一个应用程序(线程)的消息队列,由应用程序(线程)维护. 3.消息队列的关系 当鼠标.键盘产生消息时,会将消息存放到系统消息队列. 系统会根据存放的消息,找到对应窗口的消息队列.

ovs处理openflow消息的流程

ovs处理openflow消息的流程