首页 > 其他分享 >MFC:以消息为基础的事件驱动系统和消息映射机制

MFC:以消息为基础的事件驱动系统和消息映射机制

时间:2024-07-22 14:27:15浏览次数:18  
标签:控件 MFC 窗口 队列 事件驱动 线程 消息 MESSAGE

以消息为基础的事件驱动系统和消息映射机制

(1)消息

A.What(什么是消息)

本质是一个数据结构,用于应用程序不同部分之间进行通信和交互

typedef struct tagMSG {
    HWND   hwnd;        // 接收该消息的窗口句柄
    UINT   message;     // 消息标识符(如 WM_CREATE、WM_SIZE 等)
    WPARAM wParam;      // 32 位消息的特定附加信息,确切含义依赖于消息值
    LPARAM lParam;      // 32 位消息的特定附加信息,确切含义也依赖于消息值
    DWORD  time;        // 消息创建时的时间
    POINT  pt;          // 消息创建时鼠标/光标在屏幕坐标系中的位置
} MSG;
  • hwnd:表示将要接收这个消息的窗口的句柄,也就是说谁将接收到这个消息
  • message:消息的类型标识符,指定了消息的类型,也就是说明了该消息是用来干什么的
  • wParam和lParam:这两个成员用于携带消息的附加参数,其具体含义取决于具体的消息类型
  • time:消息创建时的时间
  • pt:消息创建时鼠标/光标在屏幕坐标系中的位置

B.Which(有哪些类型的消息)

  • 窗口消息:

    与窗口内部运作有关的消息,如创建窗口(wm_create)、绘制窗口(wm_paint)、鼠标移动(wm_mousemove)、控件颜色(wm_ctlcolor)、水平滚动(wm_hscroll)等

  • 命令消息:

    当用户从菜单选中一个命令项目、按下一个快捷键或者点击工具栏上的一个按钮时,都将发送 wm_command 命令消息

    BEGIN_MESSAGE_MAP(CMainFrame, CMDIFrameWnd) 
    	ON_COMMEND(IDD_BUTTON01, OnButton1LButtonDown) 
    END_MESSAGE_MAP() 
    

    上述代码表明:当点击控件IDD_BUTTON01时,将调用函数OnButton1LButtonDown()

  • 控件通知消息:

    当窗口内的子控件发生一些事件(通常由用户输入触发),需要通知父窗口时发送的消息。通知消息只适用于标准的窗口控件,如按钮、列表框、组合框、编辑框,以及 Windows 公共控件如树状视图、列表视图等

    BEGIN_MESSAGE_MAP(CMainFrame, CMDIFrameWnd) 
    	ON_NOTIFY(LVN_KEYDOWN, IDC_LIST1, OnKeyDownList1)
    END_MESSAGE_MAP() 
    
    

    当列表控件 IDC_LIST1 发送 LVN_KEYDOWN 通知消息时,OnKeyDownList1 函数就会被调用,从而可以在该函数中进行相应的处理

  • 用户定义消息:

    step01:使用#define指令定义一个消息标识符 #define WM_MYMSG WM_USER + 8
    step02:在类声明中声明消息映射 DECLARE_MESSAGE_MAP()
    step03:在类中定义消息处理函数 afx_msg LRESULT MyMsgHandler(WPARAM wParam, LPARAM lParam);
    step04:实现消息映射

    BEGIN_MESSAGE_MAP(CMainFrame, CMDIFrameWnd) 
    	ON_MESSAGE(WM_MYMSG, OnMyMsgHandler) 
    END_MESSAGE_MAP() 
    

(2)消息队列

A.What(什么是消息队列)

本质是一个存放MSG的队列数据结构,用于存放消息的队列

B.Which(MFC中有哪些消息队列)

系统消息队列:

主要接收来自硬件设备(如鼠标、键盘等)的输入消息,然后将这些消息分配到相应应用程序的线程消息队列中

线程消息队列:

存储与该线程相关的所有消息,包括系统发送的标准 Windows 消息(如 WM_CREATE、WM_SIZE 等)、用户自定义消息以及由其他线程或进程发送给本线程的消息

例如,当用户点击鼠标时,系统会产生一个鼠标点击消息,并将其放入系统消息队列。然后,Windows 会将这个消息传递到相应应用程序的线程消息队列中。应用程序的线程在消息循环中获取到这个消息后,根据消息的类型和参数进行相应的处理,比如调用相应的窗口过程函数。

C.Why(消息队列的作用)

支持事件驱动模型:

通过消息队列,应用程序能够根据不同的消息触发相应的处理逻辑,实现对用户操作和系统事件的响应

缓冲消息:

对于短时间内大量产生的消息,消息队列起到缓冲的作用,避免消息的丢失或处理混乱。它按照先进先出的原则存储消息,保证消息处理的顺序性

跨线程通信:

不同线程之间可以通过向特定线程的消息队列发送消息来进行通信,实现线程之间的协调和数据传递

D.Who(谁管理消息队列)

在MFC中,Window操作系统统一管理着所有的消息队列,包括系统消息队列和线程消息队列。

在一个多线程的应用程序中,不同线程的消息队列由操作系统统一管理,使得每个线程能够按照其自身的逻辑和优先级来处理接收到的消息,而不会出现混乱或冲突

E.How(如何使用消息队列)

  • 向消息队列中添加消息:PostMessage()和SendMessage()

    PostMessage(),将消息放入线程的消息队列后立即返回,不等待消息被处理;
    SendMessage(),直接调用窗口的窗口过程函数来处理消息,并等待处理完成后才返回

  • 从消息队列中取消息:GetMessage()和PeekMessage()

    GetMessage(),从线程的消息队列中取出一条消息,如果消息队列中没有消息,该函数会阻塞,直到有消息可用或者收到 WM_QUIT 消息
    PeekMessage(),与 GetMessage 不同的是,如果消息队列中没有消息,它不会阻塞,而是立即返回 FALSE

    消息分发的一般步骤:循环读取消息队列中的消息,并进行

    MSG msg;
    while (GetMessage(&msg, NULL, 0, 0))
    {
        TranslateMessage(&msg); //翻译消息
        DispatchMessage(&msg); //分发消息
    }
    

    TranslateMessage函数会检查传入的消息,如果发现是WM_KEYDOWN和WM_KEYUP消息的组合,且对应的键可以被键盘驱动器映射为 ASCII 字符,它就会将其转换为一条WM_CHAR消息

    DispatchMessage函数将消息分发给相应的窗口过程进行处理,因为MSG中包含消息所接收的窗口句柄

(3)消息的处理过程

外部设备或窗口内部事件产生消息,产生的消息进入线程的消息队列中,而WIndow应用程序的主函数会不断调用GetMessage函数从消息列表中获取消息,并进行消息翻译和消息转发实现消息的处理。
在这里插入图片描述

(4)消息映射机制

A.What(什么是消息映射机制)

用于将 Windows 消息与类成员函数进行关联和处理的高效机制

B.Why(消息映射机制的作用)

将这些 Windows 消息与特定的类成员函数关联起来。这样,当特定的消息被发送到窗口时,MFC 框架能够根据消息映射找到对应的处理函数,并调用它来执行相应的操作

通过消息映射机制,开发者无需直接处理复杂的 Windows 消息循环和分发逻辑,而是可以专注于编写具体的消息处理代码

C.How(如何进行消息映射)

一般而言,针对不同的消息类型,使用不同的宏进行消息映射

  • 窗口消息:

    与窗口内部运作有关的消息,如创建窗口(wm_create)、绘制窗口(wm_paint)、鼠标移动(wm_mousemove)、控件颜色(wm_ctlcolor)、水平滚动(wm_hscroll)等

  • 命令消息:

    当用户从菜单选中一个命令项目、按下一个快捷键或者点击工具栏上的一个按钮时,都将发送 wm_command 命令消息

    BEGIN_MESSAGE_MAP(CMainFrame, CMDIFrameWnd) 
    	ON_COMMEND(IDD_BUTTON01, OnButton1LButtonDown) 
    END_MESSAGE_MAP() 
    

    上述代码表明:当点击控件IDD_BUTTON01时,将调用函数OnButton1LButtonDown()

  • 控件通知消息:

    当窗口内的子控件发生一些事件(通常由用户输入触发),需要通知父窗口时发送的消息。通知消息只适用于标准的窗口控件,如按钮、列表框、组合框、编辑框,以及 Windows 公共控件如树状视图、列表视图等

    BEGIN_MESSAGE_MAP(CMainFrame, CMDIFrameWnd) 
    	ON_NOTIFY(LVN_KEYDOWN, IDC_LIST1, OnKeyDownList1)
    END_MESSAGE_MAP() 
    
    

    当列表控件 IDC_LIST1 发送 LVN_KEYDOWN 通知消息时,OnKeyDownList1 函数就会被调用,从而可以在该函数中进行相应的处理

  • 用户定义消息:

    step01:使用#define指令定义一个消息标识符 #define WM_MYMSG WM_USER + 8
    step02:在类声明中声明消息映射 DECLARE_MESSAGE_MAP()
    step03:在类中定义消息处理函数 afx_msg LRESULT MyMsgHandler(WPARAM wParam, LPARAM lParam);
    step04:实现消息映射

    BEGIN_MESSAGE_MAP(CMainFrame, CMDIFrameWnd) 
    	ON_MESSAGE(WM_MYMSG, OnMyMsgHandler) 
    END_MESSAGE_MAP() 
    

标签:控件,MFC,窗口,队列,事件驱动,线程,消息,MESSAGE
From: https://blog.csdn.net/qq_42279379/article/details/140572063

相关文章

  • Robot Operating System——借用内存型消息
    大纲功能和工作原理源码分析POD特点POD类型的优点非POD特点生成并发布“借用内存型消息”POD类型非POD类型在ROS2中,"loanedmessage"是一种消息传递机制,用于在发布者(publisher)和订阅者(subscriber)之间传递数据。它是一种高效的消息传递方式,可以避免不必要的数据......
  • CMFCToolbar 添加的工具条每次启动时位置改变
      MFC编程,添加了一个CMFCToolbar工具条,奇怪的是每次运行工具条的位置都会后移一段距离。在网上搜了一下,也没找到原因。今天浏览了一下代码,找到了原因在框架的OnCreate中开始就创建了新工具条,然后就设置了停靠位置  但是框架默认自带的工具条却在后面,重新计算了位置,......
  • rabbitmq发送消息localdatetime报错:Java 8 date/time type `java.time.LocalDateTime`
    两种解决方案:通过全局配置LocalDateTime的序列化/***json序列化增强解决Jackson序列化不了Java8日期*/@BeanpublicMessageConvertermessageConverter(){ObjectMapperom=newObjectMapper();om.setVisibility(PropertyAccessor.ALL,JsonAut......
  • P5535 【XR-3】小道消息
    原题链接题解如果k+1是质数,且n+1内没有k+1的倍数,那么只需要一天否则只需要两天如果k+1不是质数,第一天产生的质数会在第二天布满所有数如果k+1是质数,那么k+1~2k+2之间一定有一个质数,也能布满所有数实施首先要判断k是不是质数\(O(\sqrt(n))\)恰好可......
  • 华为IoTDA平台下发MQTT消息
    前一篇博文介绍了如何使用MQTTX连接华为IoTDA平台并上报消息,本文介绍一下如何下发消息。IoT设备接入平台支持MQTT协议设备进行命令下发,属性设置,属性查询以及消息下发等操作。在进入指定设备的页面后,选择“云端下发”,然后点击“下发消息”即可发送消息。 消息下发不依赖产......
  • 第三节:面向切面解决问题、本地消息表实现最终一致性、热门接口耗时长问题
    一.        二.        三.         !作       者:Yaopengfei(姚鹏飞)博客地址:http://www.cnblogs.com/yaopengfei/声     明1:如有错误,欢迎讨论,请勿谩骂^_^。声     明2:原创博客请在转载......
  • mq发送消息一般步骤
    添加依赖<!--消息发送--><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-amqp</artifactId></dependency>配置spring:rabbitmq:host:192.168.150.101#你的虚拟......
  • MQ消息处理
    消息后置处理器这个方法的目的是在消息发送之前对消息进行后处理,具体地,它向消息的属性中添加了一个名为"user-info"的头信息,这个头信息的值是通过UserContext.getUser()方法获取的当前用户信息。publicclassRelyUserInfoMessageProcessorimplementsMessagePostProcessor......
  • Spring Boot+WebSocket向前端推送消息
     ​ 博客主页:   南来_北往......
  • 深入理解淘客返利系统中的异步消息处理与队列技术
    深入理解淘客返利系统中的异步消息处理与队列技术大家好,我是微赚淘客系统3.0的小编,是个冬天不穿秋裤,天冷也要风度的程序猿!在现代的淘客返利系统中,高并发和复杂的业务需求要求我们采用异步消息处理和队列技术来提高系统的性能和可伸缩性。本文将深入探讨在淘客返利系统中如......