C#.NET 消息机制
转载:http://blog.csdn.net/fan158/article/details/6178392
一、消息概述
众人周知,window系统是一个消息驱动的系统, windows操作系统本身有自己的消息队列,消息循环,它捕捉键盘,鼠标的动作生成消息,并将这个消息传给应用程序的消息队列。 余下的工作有应用程序处理消息, windows 消息机制在这儿就不再讲述,我们重点讲述应用程序的消息机制。 大家只要明白消息是由操作系统传递给应用程序的。 一副图更能详细说明:
应用程序的执行是通过消息驱动的。消息是整个应用程序的工作引擎,我们需要理解掌握我们使用的编程语言是如何封装消息的原理。
1 什么是消息(Message)
消息就是通知和命令。在.NET框架类库中的System.Windows.Forms命名空间中微软采用面对对象的方式重新定义了Message。新的消息(Message)结构的公共部分属性基本与早期的一样,不过它是面对对象的。
公共属性:
HWnd 获取或设定消息的处理函数
Msg 获取或设定消息的ID号
Lparam 指定消息的LParam字段
Wparam 指定消息的WParam字段
Result 指定为响应消息处理函数而向OS系统返回的值
2 消息驱动的过程
所有的外部事件,如键盘输入、鼠标移动、按动鼠标都由OS系统转换成相应的消息发送到应用程序的消息队列。每个应用程序都有一段相应的程序代码来检索、分发这些消息到对应的窗体,然后由窗体的处理函数来处理。
二、C#中的消息的封装
C#对消息重新进行了面对对象的封装,在C#中消息被封装成了事件。
System.Windows.Forms.Application类具有用于启动和停止应用程序和线程以及处理Windows消息的方法。
调用Run以启动当前线程上的应用程序消息循环,并可以选择使其窗体可见。
调用Exit或ExitThread来停止消息循环。
C#中用Application类来处理消息的接收和发送的。消息的循环是由它负责的。
从本质上来讲,每个窗体一般都对应一个窗体过程处理函数。那么,C#的一个Form实例(相当于一个窗体)收到消息后是如何处理消息的?其实,这个问题的分析也就是展示了C#的消息封装原理。
实现鼠标左键按下的消息的响应(WM_LBUTTONDOWN)
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; public delegate void MouseEventHandler( object sender,MouseEventArgs e); //MouseEventHandler的定义
View Code
实际上,上面定义了一个委托类型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.Drawing; using System.Collections; using System.ComponentModel; using System.Windows.Forms; using System.Data;namespace MessageMech3 {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 class WinForm : System.Windows.Forms.Form{private System.Windows.Forms.Label label1;private System.ComponentModel.Container components = null;public WinForm(){InitializeComponent();//安装自己的过滤器 CLButtonDownFilter MyFilter = new CLButtonDownFilter();System.Windows.Forms.Application.AddMessageFilter(MyFilter);}protected override void Dispose(bool disposing){if (disposing){if (components != null){components.Dispose();}}base.Dispose(disposing);}#region Windows Form Designer generated codeprivate void InitializeComponent(){this.label1 = new System.Windows.Forms.Label();this.SuspendLayout();this.label1.BackColor = System.Drawing.Color.Transparent;this.label1.Dock = System.Windows.Forms.DockStyle.Top;this.label1.ForeColor = System.Drawing.Color.DarkViolet;this.label1.Name = "label1";this.label1.Size = new System.Drawing.Size(440, 32);this.label1.TabIndex = 0;this.label1.Text = "演示如何在App对象中处理消息,请点鼠标左键";this.label1.TextAlign = System.Drawing.ContentAlignment.BottomCenter;this.AutoScaleBaseSize = new System.Drawing.Size(7, 22);this.BackColor = System.Drawing.Color.WhiteSmoke;this.ClientSize = new System.Drawing.Size(440, 273);this.Controls.AddRange(new System.Windows.Forms.Control[] { this.label1 });this.Font = new System.Drawing.Font("华文行楷", 15F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((System.Byte)(134)));this.Name = "WinForm";this.Text = "WinForm";//消息响应函数的调用顺序和添加委托的顺序一致,即:以下命令将先调用Form1_MouseDown1再调用Form1_MouseDown2, 通过委托添加自己的鼠标按键消息响应函数1this.MouseDown += new System.Windows.Forms.MouseEventHandler(this.Form1_MouseDown1);//通过委托添加自己的鼠标按键消息响应函数2this.MouseDown += new System.Windows.Forms.MouseEventHandler(this.Form1_MouseDown2);this.ResumeLayout(false);}#endregion//应用程序的主入口点 [STAThread]static void Main(){Application.Run(new WinForm()); //启动当前Form线程上的应用程序消息循环 }// 通过C#提供的事件接口添加自己的鼠标按键事件的响应函数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函数响应");}//通过覆盖基类的事件引发函数拦截消息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);}//通过覆盖基类的窗体函数拦截消息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)System.Windows.Forms.MessageBox.Show("消息被WndProc函数响应");// WM_LBUTTONDOWN, 不需要截获消息则为base.WndProc(ref e);}} }
View Code
以上代码我们首先用类CLButtonDownFilter实现了IMessageFilter接口,在WinForm初始化的时候我们安装了消息筛选器。程序实际执行的时候,在点击鼠标左键的时候,程序仅仅会弹出一个"App中鼠标左键按下"的消息框。因为我们在消息发往窗体前拦截了它,所以窗体将接收不到WM_LBUTTONDOWN消息。
如果我们把
if(m.Msg==0x0201)// WM_LBUTTONDOWN { System.Windows.Forms.MessageBox.Show("App中鼠标左键按下"); return true;
}
改成
if (m.Msg==0x0201)// WM_LBUTTONDOWN { System.Windows.Forms.MessageBox.Show("App中鼠标左键按下"); return false;
}
那么,我们在Application类处理消息后,消息将继续发往窗体。窗体的函数将可以处理此消息。程序执行效果是顺序弹出5个消息框。
1:<<App中鼠标左键按下>>
2:<<消息被WndProc函数响应>>
3:<<消息被OnMouseDown函数响应>>
4:<<消息被Form1_MouseDown1函数响应>>
5:<<消息被Form1_MouseDown2函数响应>>
其实本文中已经说的挺详细的.弹出的对话框只是为了让你更直观的看出导致的结果.
先定义没过滤时的效果.
this.MouseDown += new System.Windows.Forms.MouseEventHandler(this.Form1_MouseDown1);
private void Form1_MouseDown1(object sender, System.Windows.Forms.MouseEventArgs e)
{
if (e.Button == System.Windows.Forms.MouseButtons.Left)
MessageBox.Show("消息被Form1_MouseDown1函数响应");
}
主要有两种方法过滤实现过滤
第一种:
protected override void WndProc(ref Message m)
{ if (m.Msg == 0x0201) return; else base.WndProc(ref m);
}
第二种
不重写WndProc
//实现消息过滤器接口
public class CLButtonDownFilter : IMessageFilter
{
public bool PreFilterMessage(ref Message m)
{
if (m.Msg == 0x0201)// WM_LBUTTONDOWN
{ //返回值为true, 表示消息已被处理,不要再往后传递,因此消息被截获 //返回值为false,表示消息未被处理,需要再往后传递,因此消息未被截获 return true;
}
return false;
}
} CLButtonDownFilter MyFilter = new CLButtonDownFilter();
System.Windows.Forms.Application.AddMessageFilter(MyFilter);
转载于:https://www.cnblogs.com/HouZhiHouJueBlogs/articles/3841001.html
C#.NET 消息机制相关推荐
- Android消息机制学习笔记
Android的消息机制主要是指Handler的运行机制,Handler的运行需要底层的MessageQueue和Looper的支撑: MessageQueue:消息队列,它的内存存储了一组消息,以队 ...
- WPF的消息机制(二)- WPF内部的5个窗口之隐藏消息窗口
原文:WPF的消息机制(二)- WPF内部的5个窗口之隐藏消息窗口 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/powertoolsteam/ar ...
- iOS开发系列--通知与消息机制
http://www.cocoachina.com/ios/20150318/11364.html 概述 在多数移动应用中任何时候都只能有一个应用程序处于活跃状态,如果其他应用此刻发生了一些用户感兴趣 ...
- Android消息机制Handler用法
这篇文章介绍了Android消息机制Handler用法总结,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧 1.简述 Handler消息机制主要包括: Messa ...
- EJB与JAVA BEAN_J2EE的异步消息机制
EJB与JAVA BEAN_J2EE的异步消息机制 EJB与JAVA BEAN的区别 Java Bean 是可复用的组件,对Java Bean并没有严格的规范,理论上讲,任何一个Java类都可以是一个 ...
- Android异步消息机制
2019独角兽企业重金招聘Python工程师标准>>> 目录介绍 1.Handler的常见的使用方式 2.如何在子线程中定义Handler 3.主线程如何自动调用Looper.pre ...
- Android的消息机制(2)
上一节中,是主线程自己发了一个消息到自己的Message Queue中,并把消息从队列中提取出来.那么如何由别的线程发送消息给主线程的Message Queue中呢? 直接看代码~~ 1 2 3 4 ...
- 【腾讯Bugly干货分享】经典随机Crash之二:Android消息机制
为什么80%的码农都做不了架构师?>>> 本文作者:鲁可--腾讯SNG专项测试组 测试工程师 背景 承上经典随机Crash之一:线程安全 问题的模型 好几次灰度top1.top ...
- windows程序消息机制(Winform界面更新有关)--转
1. Windows程序消息机制 Windows GUI程序是基于消息机制的,有个主线程维护着消息泵.这个消息泵让windows程序生生不息. Windows程序有个消息队列,窗体上的所有消息是这个队 ...
- CCBPM工作流引擎的消息机制与设计
为什么80%的码农都做不了架构师?>>> CCBPM工作流引擎的消息机制与设计 关键字:ccflow jflow 消息机制 流程引擎 自动发送短信 发送邮件 发送消息 流程引擎 ...
最新文章
- 在Windows 8.1上使用Fiddler重定向http请求进行前端调试
- 如何看待机器视觉的“对抗样本”问题,其原理是什么?
- 2020科目一考试口诀_科目一考试口诀,看完少练超多题,需要轻松备考的学员请收藏!...
- Sentry UDFWhiteList bug分析
- 程序员修神之路--略懂数据库集群读写分离而已
- matlab宏参赛,MATLAB杯无人机大赛 | 决赛通知!
- arcgis中dem坐标定义_GIS基础-DEM Grid规则格网结构
- element ui实现动态显示textarea剩余字数
- idea弹出Server‘s certificate is not trusted
- PC网站接入微信登陆流程一:微信开放平台账号注册和开发者资质认证
- 内存(主存)(一般指电脑内存条)包含RAM(SRAM,DRAM),ROM,高速缓存(CACHE),SDRAM,DDRRAM
- MATLAB 显示和保存傅里叶变化频谱图
- mysql root 访问被拒绝_mysql-“连接失败:用户'root'@'localhost'(使用密码:是)的访问被拒绝”...
- QT MPlayer播放器
- 如何查看Excel文件最大行数
- 电子科技大学信息与软件工程学院
- 守护游戏安全,保障开发权益丨客户案例
- 019动物识别检测网页版
- 【女装2018新款潮碎花半身裙韩版气质两件套裙子】http://m.tb.cn/h.3aHr1L7
- STM32CubeMX | 31-使用硬件FMC读写SDRAM(W9825G6KH)