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

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

  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的值:

  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完成对消息的处理。

  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. };

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

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

    原文网址:http://blog.csdn.net/hyhnoproblem/article/details/6182585 该示例通过研究基本的单文档程序的“文件”--“打开”命令,分析WM_COM ...

  2. Windows窗口消息大全(转)

    Windows窗口消息大全,全不全自己看 ////////////////////////////////////////////////////////////////////////// #inc ...

  3. Windows窗口消息大全

    ////////////////////////////////////////////////////////////////////////// #include "AFXPRIV.H& ...

  4. Windows程序设计--(三)窗口与消息

    3.1 窗口的创建 3.1.1 系统结构概述 所谓「Windows给程序发送消息」,是指Windows呼叫程序中的一个函数,该函数的参数描述了这个特定消息.这种位于Windows程序中的函数称为「窗口 ...

  5. [8]windows内核情景分析--窗口消息

    消息与钩子 众所周知,Windows系统是消息驱动的,现在我们就来看Windows的消息机制. 早期的Windows的窗口图形机制是在用户空间实现的,后来为了提高图形处理效率,将这部分移入内核空间,在 ...

  6. 关于 OnCloseQuery: 顺序、不能关机等(所有的windows的广播消息都是逐窗口传递的)——如果一个窗体的OnCloseQuery事件中如果写了代码那么WM_QUERYENDSESSION消息就传不过去了msg.result会返回0,关机事件也就停止了

    系统关闭窗体的事件顺序为: OnCloseQuery ----> OnClose ----> OnDestroy 下面的代码说明问题: unit Unit3; interface uses ...

  7. 深刻:截获windows的消息并分析实例(DefWindowProc),以WM_NCHITTEST举例(Windows下每一个鼠标消息都是由 WM_NCHITTEST 消息产生的,这个消息的参数包含了鼠标位置的信息)

    1,回调函数工作机制 回调函数由操作系统自动调用,回调函数的返回值当然也是返回给操作系统了. 2,截获操作系统发出的消息,截获到后,将另外一个消息返回给操作系统,已达到欺骗操作系统的目的. 下面还是以 ...

  8. Windows消息理解(系统消息队列,进程消息队列,非队列消息)

    // ====================Windows消息分类==========================在Windows中,消息分为以下三类:标准消息——除WM_COMMAND之外,所 ...

  9. [转] C#中发送消息给指定的窗口,以及接收消息

    原文C#中发送消息给指定的窗口,以及接收消息 public class Note { //声明 API 函数 [DllImport("User32.dll", EntryPoint ...

随机推荐

  1. 图像处理(二十一)基于数据驱动的人脸卡通动画生成-Siggraph Asia 2014

    http://blog.csdn.net/garfielder007/article/details/50582018 在现实生活中,我们经常会去评价一个人,长得是否漂亮.是不是帅哥美女,然而如何用五 ...

  2. java第六天

    p37 1.java ant详解 练习8 /** * Created by xkfx on 2017/2/26. */ class A { static int i = 47; } public cl ...

  3. mysql中生成列与JSON类型的索引

    MySQL中支持生成列,生成列的值是根据列定义中包含的表达式计算的. 一个简单的例子来认识生成列! CREATE TABLE triangle( sidea DOUBLE, sideb DOUBLE, ...

  4. 20145105 《Java程序设计》实验一总结

    实验一   Java开发环境的熟悉 一.    实验内容: (一)使用JDK编译.运行简单的程序 (二)使用idea编辑.编译.运行.调试Java程序. 二.    实验步骤: (一)   命令行下J ...

  5. 线程访问ui,托管

    1.在类中声明 delegate void setDebugDelegate(string info);//线程访问textbox委托函数 private void setDebug(string i ...

  6. 日志自定义Tag

    import java.util.concurrent.ConcurrentMap; import java.util.concurrent.ConcurrentHashMap; /** * Crea ...

  7. [微信开发] - weixin4j关键类解析

    TokenUtil : get()获取我方自定义的token(从配置文件或数据库) checkSignature(Str..... (服务器配置连接验证有效性) /* * 微信公众平台(JAVA) S ...

  8. python爬虫scrapy学习之篇二

    继上篇<python之urllib2简单解析HTML页面>之后学习使用Python比较有名的爬虫scrapy.网上搜到两篇相应的文档,一篇是较早版本的中文文档Scrapy 0.24 文档, ...

  9. Codeforces Round #381 (Div. 2) D. Alyona and a tree dfs序+树状数组

    D. Alyona and a tree time limit per test 2 seconds memory limit per test 256 megabytes input standar ...

  10. 反射中的 Class.forName() 与 ClassLoader.loadClass() 的区别

    在Java中,类加载器把一个类加载进Java虚拟机中,要经过三个步骤来完成:加载.链接和初始化,其中链接又可以分成验证.准备和解析三步,除了解析外,其它步骤是严格按照顺序完成的,各个步骤的主要工作如下 ...