转自原文 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_LBUTTONDOWNSystem.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,

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

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

本文参考源码 MSGFilter.rar

没有整理与归纳的知识,一文不值!高度概括与梳理的知识,才是自己真正的知识与技能。 永远不要让自己的自由、好奇、充满创造力的想法被现实的框架所束缚,让创造力自由成长吧! 多花时间,关心他(她)人,正如别人所关心你的。理想的腾飞与实现,没有别人的支持与帮助,是万万不能的。
本文转自wenglabs博客园博客,原文链接:http://www.cnblogs.com/arxive/p/7683761.html,如需转载请自行联系原作者

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

  1. Android多线程----异步消息处理机制之Handler

    虽然是国庆佳节,但也不能停止学习的脚步,我选择在教研室为祖国母亲默默地庆生. 关于Android的多线程知识,请参考本人之前的一篇博客:Android 多线程----AsyncTask异步任务详解 在 ...

  2. Android消息处理机制

    Google参考了Windows的消息处理机制,在Android系统中实现了一套类似的消息处理机制.学习Android的消息处理机制,有几个概念(类)必须了解: 1.       Message 消息 ...

  3. Android 应用程序消息处理机制(Looper、Handler)分析

    Android应用程序是通过消息来驱动的,系统为每一个应用程序维护一个消息队例,应用程序的主线程不断地从这个消息队例中获取消息(Looper),然后对这些消息进行处理(Handler),这样就实现了通 ...

  4. qt 消息处理机制与window消息处理机制的比较

    一:windows程序的消息处理  windows程序的处理大概一致 如下: 1.0 windows 消息处理机制: int WINAPI WinMain(HINSTANCE hInstance, H ...

  5. Android应用程序消息处理机制(Looper、Handler)分析

    文章转载至CSDN社区罗升阳的安卓之旅,原文地址:http://blog.csdn.net/luoshengyang/article/details/6817933 Android应用程序是通过消息来 ...

  6. Android开发之异步消息处理机制AsyncTask

    转自:Android AsyncTask完全解析,带你从源码的角度彻底理解 另外一篇比较详细的博文:http://blog.csdn.net/liuhe688/article/details/6532 ...

  7. 认识Android(常用布局,控件,四大组件,动画,自定义控件及异常消息处理机制)

    目录 一.布局 1.LinearLayout(线性布局): 2.相对布局(RelativeLayout) 3.GridLayout(网格布局) 4.FrameLayout(帧布局): 二.控件 1.T ...

  8. 张赐荣 | Windows 消息处理机制 入门

    张赐荣 | Windows 消息处理机制 入门 [文 / 张赐荣] 事件驱动和消息循环 消息概述 消息,就是指Windows发出的一个通知,告诉应用程序某个事情发生了.例如,单击鼠标.改变窗口尺寸.按 ...

  9. Android Handler 异步消息处理机制的妙用 创建强大的图片载入类

    转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/38476887 ,本文出自[张鸿洋的博客] 近期创建了一个群.方便大家交流,群号: ...

最新文章

  1. 机器学习PAL产品优势
  2. Tomcat7 配置 ssl
  3. Halcon算子:min_max_gray和gray_histo的区别
  4. 3分钟学会Mysql查询之表连接,左连接,右连接,内连接…
  5. Dataset之CIFAR-10:CIFAR-10数据集简介、下载、使用方法之详细攻略
  6. linux编译安装madam,linux 下 使用 mdadm 创建阵列
  7. java quartz 数据库_SpringBoot+Quartz+数据库存储
  8. 阿里如何将“高峰前扩容、高峰后缩容”的梦想照进现实?
  9. ubuntu server设置时区和更新时间
  10. sqlserver object_id()函数
  11. a=a++问题引发的思考
  12. 计算机生成目录步骤word,word生成目录步骤,word怎样做目录
  13. [渝粤教育] 云南大学 大学生心理健康教育 参考 资料
  14. css使用clac()垂直居中
  15. Ubuntu 搜狗输入法 关闭简繁切换快捷键
  16. 我是一个将近30岁的女程序员
  17. U盘拷贝4G以上的文件(非格式化)
  18. ip地址合不合法怎么看_怎样判断IP地址的非法性?
  19. 全国计算机等级考试一级大纲,全国计算机等级考试大纲(一级WIN)
  20. Ionic 学习笔记之-痛彻心扉的环境搭建

热门文章

  1. 重写Java Object类中的equals()方法
  2. mysql中数据类型总结_mysql数据类型总结
  3. python学习笔记(九)——文件和异常(重点)
  4. c语言 二进制输出_程序员入门C语言,需要掌握的4个基础知识
  5. LtRecyclerView:自带上拉下拉,能增加头条目和尾条目的RecyclerView
  6. 树莓派:在ubuntu20-server安装和卸载桌面
  7. 2020-12-11 Python中的 if __name__ == “__main__“
  8. OSError: [WinError 126] 找不到指定的模块————Shapely
  9. java回退一格_api短信接口
  10. java 垃圾回收 指针_Java 和 C++ 垃圾回收之间的差别