我的编程空间,编程开发者的网络收藏夹
学习永远不晚

C#如何对消息进行封装

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

C#如何对消息进行封装

这篇文章给大家分享的是有关C#如何对消息进行封装的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。

一、C#消息概述

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;

而MouseEventHandler的定义为:

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.Drawing;   using System.Collections;   using System.ComponentModel;   using System.Windows.Forms;   using System.Data;    namespace MessageMech4   {       //实现消息过滤器接口      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;         }      }        /// < summary>      /// Summary description for WinForm.      /// < /summary>      public class WinForm : System.Windows.Forms.Form      {         /// < summary>         /// Required designer variable.         /// < /summary>         private System.Windows.Forms.Label label1;         private System.ComponentModel.Container components = null;          public WinForm()         {            //            // Required for Windows Form Designer support            //            InitializeComponent();             //            // TODO: Add any constructor code after InitializeComponent call            //            //安装自己的过滤器            CLButtonDownFilter MyFilter=new CLButtonDownFilter();            System.Windows.Forms.Application.AddMessageFilter(MyFilter);         }          /// < summary>         /// Clean up any resources being used.         /// < /summary>         protected override void Dispose (bool disposing)         {            if (disposing)            {               if (components != null)               {                  components.Dispose();               }            }            base.Dispose(disposing);         }          #region Windows Form Designer generated code         /// < summary>         /// Required method for Designer support - do not modify         /// the contents of this method with the code editor.         /// < /summary>         private void InitializeComponent()         {            this.label1 = new System.Windows.Forms.Label();            this.SuspendLayout();            //             // label1            //             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;            //             // Form1            //             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             //通过委托添加自己的鼠标按键消息响应函数1            this.MouseDown += new System.Windows.Forms.MouseEventHandler(this.Form1_MouseDown1);            //通过委托添加自己的鼠标按键消息响应函数2            this.MouseDown += new System.Windows.Forms.MouseEventHandler(this.Form1_MouseDown2);             this.ResumeLayout(false);         }         #endregion          /// < summary>         /// 应用程序的主入口点。         /// < /summary>         [STAThread]         static void Main()          {            Application.Run(new WinForm()); //启动当前Form线程上的应用程序消息循环         }          //要点1         // 通过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函数响应");          }         //要点2         //通过覆盖基类的事件引发函数拦截消息         //         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);         }         //要点3         //通过覆盖基类的窗体函数拦截消息         //         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);         }       }   }

以上代码我们首先用类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个消息框。

< < App中鼠标左键按下>>

< < 消息被WndProc函数响应>>

< < 消息被OnMouseDown函数响应>>

< < 消息被Form1_MouseDown1函数响应>>

< < 消息被Form1_MouseDown2函数响应>>

主要有两种方法过滤实现过滤

***种:

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);

感谢各位的阅读!关于“C#如何对消息进行封装”这篇文章就分享到这里了,希望以上内容可以对大家有一定的帮助,让大家可以学到更多知识,如果觉得文章不错,可以把它分享出去让更多的人看到吧!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

C#如何对消息进行封装

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

C#如何对消息进行封装

这篇文章给大家分享的是有关C#如何对消息进行封装的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。一、C#消息概述 Windows下应用程序的执行是通过消息驱动的。消息是整个应用程序的工作引擎,我们需要理解掌握我们使
2023-06-17

C++ protobuf中如何对不同消息内容进行赋值

今天小编给大家分享一下C++ protobuf中如何对不同消息内容进行赋值的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。本文
2023-07-05

如何使用TS对axios的进行简单封装

在vue项目中和后台交互获取数据这块,我们通常使用的是axios库,这篇文章主要给大家介绍了关于如何使用TS对axios的进行简单封装的相关资料,需要的朋友可以参考下
2022-11-13

JS项目中如何对本地存储进行二次封装

今天小编给大家分享一下JS项目中如何对本地存储进行二次封装的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。前言平时在开发的中,
2023-07-02

小程序中如何对网络请求进行二次封装

这篇文章主要介绍了小程序中如何对网络请求进行二次封装,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。1、背景在做微信小程序开发的时候难免会涉及到网络请求操作,小程序提供的原生网
2023-06-25

如何进行SpringBoot+RabbitMQ方式收发消息

本篇文章给大家分享的是有关如何进行SpringBoot+RabbitMQ方式收发消息,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。本篇会和SpringBoot做整合,采用自动配
2023-06-16

如何在java项目组中进行封装

今天就跟大家聊聊有关如何在java项目组中进行封装,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。java 中的封装介绍及使用方法在面向对象程式设计方法中,封装(英语:Encapsul
2023-05-31

如何在Java项目中利用post请求对Json进行封装

如何在Java项目中利用post请求对Json进行封装?针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。直接上代码String longUrl = "https://open.
2023-05-31

如何对C/C++头文件进行说明

这期内容当中小编将会给大家带来有关如何对C/C++头文件进行说明,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。C、传统 C++#include //设定插入点#include
2023-06-17

如何用C#对[0,100]进行匹配

这篇文章主要讲解了“如何用C#对[0,100]进行匹配”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“如何用C#对[0,100]进行匹配”吧!下面是一个比较综合的示例,对于匹配[0,100],
2023-06-17

C#如何利用RabbitMQ实现点对点消息传输

这篇文章主要介绍C#如何利用RabbitMQ实现点对点消息传输,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!消息队列模型所有 MQ 产品从模型抽象上来说都是一样的过程:消费者(consumer)订阅某个队列。生产者(
2023-06-15

C++程序设计如何建立对象间消息连接

本篇内容主要讲解“C++程序设计如何建立对象间消息连接”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“C++程序设计如何建立对象间消息连接”吧!一、回调类的数据结构及其成员函数本文提出的CallB
2023-06-17

如何对C++链表进行解读分析

如何对C++链表进行解读分析,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。C++语言是学习数据结构的很好的学习工具,能够全面的理解了C++中C++链表的作用和用途,那么对于理解
2023-06-17

C#如何使用NPOI对word进行读写

这篇文章主要介绍了C#如何使用NPOI对word进行读写的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇C#如何使用NPOI对word进行读写文章都会有所收获,下面我们一起来看看吧。一、简介操作Word的类库:N
2023-07-02

工具 | 如何对 MySQL 进行 TPC-C 测试?

作者:丁源 RadonDB 测试负责人负责 RadonDB 云数据库、容器化数据库的质量性能测试,迭代验证。对包括云数据库以及容器化数据库性能和高可用方案有深入研究。|背景根据 DWorks 2020 年发布的《中国自研数据库登顶 TPC-C 的意义》[1]
工具 | 如何对 MySQL 进行 TPC-C 测试?
2015-03-18

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录