C# 消息处理机制及自定义过滤方式

摘要:
Application类在C#中用于处理消息的接收和发送。它负责消息的循环。C#的Form实例在收到消息后如何处理消息?事实上,对这个问题的分析表明了C#的消息封装原理。捕获并响应事件的对象称为事件接收器。因此,C#中的事件字段本质上是一个函数指针列表,用于在某些消息到达时维护响应函数的地址。

一、消息概述
Windows 下应用程序的执行是通过消息驱动的。消息是整个应用程序的工作引擎,我们需要理解掌握我们使用的编程语言是如何封装消息的原理。
1. 什么是消息(Message)
消息就是通知和命令。在.NET框架类库中的System.Windows.Forms命名空间中微软采用面对对象的方式重新定义了Message。新的消息(Message)结构的公共部分属性基本与早期的一样,不过它是面对对象的。
公共属性:

public IntPtr HWnd { get; set; }获取或设置消息的窗口句柄
public int Msg { get; set; }获取或设置消息的 ID 号
public IntPtr Result { get; set; }指定为响应消息处理而向 Windows 返回的值
public IntPtr LParam { get; set; }指定消息的 System.Windows.Forms.Message.LParam 字段
public IntPtr WParam { get; set; }获取或设置消息的 System.Windows.Forms.Message.WParam 字段

2. 消息驱动的过程
所有的外部事件,如键盘输入、鼠标移动、按动鼠标都由OS系统转换成相应的消息发送到应用程序的消息队列。每个应用程序都有一段相应的程序代码来检索、分发这些消息到对应的窗体,然后由窗体的处理函数来处理。

二、C#中的消息的封装
C#对消息重新进行了面对对象的封装,在C#中消息被封装成了事件。System.Windows.Forms.Application 类具有用于启动和停止应用程序和线程以及处理Windows消息的方法。

调用Run以启动当前线程上的应用程序消息循环,并可以选择使其窗体可见。
调用Exit或ExitThread来停止消息循环。
C#中用Application类来处理消息的接收和发送。消息的循环是由它负责的。
从本质上来讲,每个窗体一般都对应一个窗体过程处理函数。那么,C#的一个Form实例(相当于一个窗体)收到消息后是如何处理消息的?其实,这个问题的分析也就是展示了C#的消息封装原理。
实现鼠标左键按下的消息的响应(WM_LBUTTONDOWN)

public partial class Form1 : Form
{
    public Form1()
    {
        InitializeComponent();
        this.MouseDown += new System.Windows.Forms.MouseEventHandler(this.Form1_MouseDown1);
        this.MouseDown += new System.Windows.Forms.MouseEventHandler(this.Form1_MouseDown2);
    }
    private void Form1_MouseDown1(object sender, System.Windows.Forms.MouseEventArgs e)
    {
        if (e.Button == System.Windows.Forms.MouseButtons.Left)
            System.Windows.Forms.MessageBox.Show("消息被Form1_MouseDown1函数响应");
    }
    private void Form1_MouseDown2(object sender, System.Windows.Forms.MouseEventArgs e)
    {
        if (e.Button == System.Windows.Forms.MouseButtons.Left)
            System.Windows.Forms.MessageBox.Show("消息被Form1_MouseDown2函数响应");
    }
}

上面 this.MouseDown 是C#中的一个事件。它的定义如下:

//
// 摘要:
//     当鼠标指针位于控件上并按下鼠标键时发生。
public event MouseEventHandler MouseDown;
// 摘要:
//     表示将处理窗体、控件或其他组件的 MouseDown、MouseUp 或 MouseMove 事件的方法。
//
// 参数:
//   sender:
//     事件源。
//
//   e:
//     包含事件数据的 System.Windows.Forms.MouseEventArgs。
public delegate void MouseEventHandler(object sender, MouseEventArgs e);

实际上,上面定义了一个委托类型 MouseEventHandler。委托了启用了其它编程语言中的函数指针的解决方案。与C++的函数指针不同,委托是完全面向对象的,同时封装了对象实例和方法。本质上,委托把一个实例和该实例上的方法函数封装成一个可调用的实体,它是面对对象的、安全的。
我们可以把 this.MouseDown += new System.Windows.Forms.MouseEventHandler(this.Form1_MouseDown1); 这条语句看成向 this.MouseDown 添加一个函数指针。

事件是对象发送的消息,以发送信号通知操作的发生。引发(触发)事件的对象叫做事件发送方。捕获事件并对事件作出响应的对象叫做事件接收方。在事件通讯中,事件发送方类并不知道哪个对象或方法将接收到(处理)它引发的事件。所需要的是在发送方和接收方之间存在一个媒介(类似指针的机制)。.NET框架定义了一个特殊的类型(Delegate委托),该类型提供函数指针的功能。这样,委托就等效于一个类型安全的函数指针或一个回调函数。

前面我们向this.MouseDown事件添加了两个委托。
this.MouseDown += new System.Windows.Forms.MouseEventHandler(this.Form1_MouseDown1);
this.MouseDown += new System.Windows.Forms.MouseEventHandler(this.Form1_MouseDown2);
结果,我们的两个函数 Form1_MouseDown1、Form1_MouseDown2 在我们单击鼠标左键的时候都会被调用,而且调用的顺序和我们添加委托的顺序一致。
WM_LBUTTONDOWN消息首先被Application类从应用程序消息队列中取出,然后分发到相应的窗体。窗体使用MouseDown事件中的函数指针调用已经添加的响应函数。所以, C# 中的事件字段实质上是一个函数指针列表,用来维护一些消息到达时的响应函数的地址。

三、结论
C# 中消息的工作流程:
C# 中的消息被Application类从应用程序消息队列中取出,然后分发到消息对应的窗体,窗体对象的第一个响应函数是对象中的 protected override void WndProc(ref System.Windows.Forms.Message e)方法。
它再根据消息的类型调用默认的消息响应函数(如OnMouseDown),默认的响应函数然后根据对象的事件字段(如this.MouseDown )中的函数指针列表,调用用户所加入的响应函数(如Form1_MouseDown1和Form1_MouseDown2),而且调用顺序和用户添加顺序一致。

四、再回首 Application 类
Application 类有一个 AddMessageFilter 的静态方法,通过它我们可以添加消息筛选器,以便在向目标传递Windows消息时,检视这些消息。使用消息筛选器来防止引发特定事件,或在将某事件传递给事件处理程序之前使用消息筛选器对其执行特殊操作。我们必须提供 IMessageFilter 接口的一个实现,然后才可以使用消息筛选器。以下的示范代码将演示在消息发往窗体前我们如何拦截它。我们拦截的同样是WM_LBUTTONDOWN消息。

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
using System.Threading;
namespace WindowsFormsApplication1
{
    /// <summary>
    /// 实现消息过滤器接口
    /// </summary>
    public class CLButtonDownFilter : IMessageFilter
    {
        public bool PreFilterMessage(ref Message m)
        {
            if (m.Msg == 0x0201) // WM_LBUTTONDOWN
            {
                System.Windows.Forms.MessageBox.Show("App中鼠标左键按下");
                // 返回值为true, 表示消息已被处理,不要再往后传递,因此消息被截获
                // 返回值为false,表示消息未被处理,需要再往后传递,因此消息未被截获
                return true;
            }
            return false;
        }
    }
    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
            // 安装自定义消息过滤器
            CLButtonDownFilter MyFilter = new CLButtonDownFilter();
            Application.AddMessageFilter(MyFilter);
            this.MouseDown += new System.Windows.Forms.MouseEventHandler(this.Form1_MouseDown1);
            this.MouseDown += new System.Windows.Forms.MouseEventHandler(this.Form1_MouseDown2);
        }
        private void Form1_MouseDown1(object sender, System.Windows.Forms.MouseEventArgs e)
        {
            if (e.Button == System.Windows.Forms.MouseButtons.Left)
                System.Windows.Forms.MessageBox.Show("消息被Form1_MouseDown1函数响应");
        }
        private void Form1_MouseDown2(object sender, System.Windows.Forms.MouseEventArgs e)
        {
            if (e.Button == System.Windows.Forms.MouseButtons.Left)
                System.Windows.Forms.MessageBox.Show("消息被Form1_MouseDown2函数响应");
        }
        /// <summary>
        /// 通过覆盖基类的窗体函数拦截消息
        /// </summary>
        /// <param name="e"></param>
        protected override void WndProc(ref System.Windows.Forms.Message e)
        {
            //如果需要截获消息,
            //if(e.Msg==0x0201)// WM_LBUTTONDOWN
            // System.Windows.Forms.MessageBox.Show("消息被WndProc函数响应");
            //else
            // base.WndProc(ref e);
            //不需要截获消息则为
            if (e.Msg == 0x0201)// WM_LBUTTONDOWN
                System.Windows.Forms.MessageBox.Show("消息被WndProc函数响应");
            base.WndProc(ref e);
        }
        /// <summary>
        /// 通过覆盖基类的事件引发函数拦截消息
        /// </summary>
        /// <param name="e"></param>
        protected override void OnMouseDown(MouseEventArgs e)
        {
            if (e.Button == System.Windows.Forms.MouseButtons.Left)
                System.Windows.Forms.MessageBox.Show("消息被OnMouseDown函数响应");
            //如果需要截获消息,可将base.OnMouseDown(e);语句注释掉
            base.OnMouseDown(e);
        }
    }
}

以上代码我们首先用类 CLButtonDownFilter 实现了 IMessageFilter 接口,在WinForm初始化的时候我们安装了消息筛选器。程序实际执行的时候,在点击鼠标左键的时候,程序仅仅会弹出一个"App中鼠标左键按下"的消息框。因为我们在消息发往窗体前拦截了它,所以窗体将接收不到WM_LBUTTONDOWN消息。
如果我们把这个代码块改为返回 false,

image
那么,我们在Application类处理消息后,消息将继续发往窗体。窗体的函数将可以处理此消息。程序执行效果是顺序弹出 5 个消息框。
1:<<App中鼠标左键按下>>
2:<<消息被WndProc函数响应>>
3:<<消息被OnMouseDown函数响应>>
4:<<消息被Form1_MouseDown1函数响应>>
5:<<消息被Form1_MouseDown2函数响应>>

其实本文中已经说的挺详细的.弹出的对话框只是为了让你更直观的看出导致的结果.

免责声明:文章转载自《C# 消息处理机制及自定义过滤方式》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇spring boot集成akka常用EXE文件反编译工具【转】下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

ZAB协议简介

Zookeeper 使用 Zookeeper Atomic Broadcast (ZAB) 协议来保障分布式数据一致性。 ZAB是一种支持崩溃恢复的消息广播协议,采用类似2PC的广播模式保证正常运行时性能,并使用基于 Paxos 的策略保证崩溃恢复时的一致性。 在阅读本文前建议先了解2PC和Paxos ZAB协议中节点存在四种状态: Leading: 当...

RocketMQ学习分享

消息队列的流派   什么是 MQ Message Queue(MQ),消息队列中间件。很多人都说:MQ 通过将消息的发送和接收分离来实现应用程序的异步和解偶,这个给人的直觉是——MQ 是异步的,用来解耦的,但是这个只是 MQ 的效果而不是目的。MQ 真正的目的是为了通讯,屏蔽底层复杂的通讯协议,定义了一套应用层的、更加简单的通讯协议。一个分布式系统中两个模...

关于Java高并发的问题

前言: 对于开发的网站,如果网站的访问量非常大的话,那么就需要考虑相关的并发访问问题了。 一.同步和异步的区别和联系 所谓同步,可以理解为在执行完一个函数或方法之后,一直等待系统返回值或消息,这时程序是出于阻塞的,只有接收到返回的值或消息后才往下执行其它的命令。 异步,执行完函数或方法后,不必阻塞性地等待返回值或消息,只需要向系统委托一个异步过程,那么当系...

Samba 简介

SMB 代表的是服务器消息块 (Server Message Block),它是用于在 Windows 上共享文件的协议的原始名称。 CIFS 代表公共 Internet 文件系统 (Common Internet File System),它是 Microsoft 描述该协议最近一个版本的新字首组合词。 samba认证  Samba 有它自己独特的口令数...

nonebot 源码阅读笔记

前言 nonebot 是一个 QQ 消息机器人框架,它的一些实现机制,值得参考。 nonebot NoneBot 初始化(配置加载) 阅读 nonebot 文档,第一个示例如下: import nonebot if __name__ == '__main__': nonebot.init() nonebot.load_builtin_pl...

10 TCP限流技术

  TCP限流是因为让接收方充分接受完消息,保证数据安全,不会丢失 一、窗口机制介绍 发送端和接收端都拥有一个窗口,当发送端发送数据时,落进窗口的数据被发送,当接受端接受数据时,落进接收端窗口的数据将会被接受。由此可见可以通过窗口大小限制流量 二、滑动窗口技术(限流使用到的技术) 1 TCP滑动窗口技术通过动态改变窗口大小来调节两台主机间的数据传输。每个T...