夜鹰教程网-程序员的加油站
 当前位置:文章中心 >> vs2022_vs2019_vs2017_vs2014_vs2012
Windows下应用程序的执行是通过消息驱动的
夜鹰教程网 来源:www.yyjcw.com 日期:2016-11-1 15:11:29
消息是整个应用程序的工作引擎,我们需要理解掌握我们使用的编程语言是如何封装消息的原理。 

一、消息概述

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


   /// <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个消息框。
    1:<<App中鼠标左键按下>>
    2:<<消息被WndProc函数响应>>
    3:<<消息被OnMouseDown函数响应>>
    4:<<消息被Form1_MouseDown1函数响应>>
    5:<<消息被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);

 

本文来自于《电脑编程技巧与维护》2002年7期。

出处:http://blog.csdn.net/acupofnescafe/archive/2009/01/13/3768944.aspx

复制链接 网友评论 收藏本文 关闭此页
上一条: Microsoft Visual Studio 2012 Ul…  下一条: SQL_Server_2005数据库实践教程—5.第五…
夜鹰教程网成立于2008年,目前已经运营了将近 13 年,发布了大量关于 html5/css3/C#/asp.net/java/python/nodejs/mongodb/sql server/android/javascript/mysql/mvc/easyui/vue/echarts原创教程。 我们一直都在坚持的是:认证负责、一丝不苟、以工匠的精神来打磨每一套教程,让读者感受到作者的用心。我们默默投入的时间,确保每一套教程都是一件作品,而不是呆板的文字和视频! 目前我们推出在线辅导班试运营,模式为一对一辅导,教学工具为QQ。我们的辅导学科包括 java 、android原生开发、webapp开发、商城开发、C#和asp.net开发,winform和物联网开发、web前端开发,但不仅限于此。 普通班针对的是国内学员,例如想打好基础的大学生、想转行的有志青年、想深入学习的程序员、想开发软件的初学者或者业余爱好者等。 就业办针对即将毕业上岗的大四学生,或者打算转行的初级开发工程师。 留学生班针对的是在欧美、加拿大、澳洲、日本、韩国、新加坡等地留学的中国学子,目的是让大家熟练地掌握编程技能,按时完成老师布置的作业,并能顺利地通过考试。 详细咨询QQ:1416759661   夜鹰教程网  基于角色的权限管理系统(c-s/b-s)。
  夜鹰教程网  基于nodejs的聊天室开发视频教程
  夜鹰教程网  Git分布式版本管理视频教程
  夜鹰教程网  MVC+EasyUI视频教程
  夜鹰教程网  在线考试系统视频教程
  夜鹰教程网  MongoDB视频教程。
  夜鹰教程网 Canvas视频教程
  夜鹰教程网 报表开发视频教程
  热点推荐
一个关于天气预报的WebService【C…
VS2010最大的新特点是并行编程的进…
TextBox控件:asp.net中如何为密码…
Web服务调用实例:实现天气预报的…
ASP.NET程序员面试试题(130道题)
ASP.NET教程:调用WebService的源码…
网站开发全程设计
据说这套.net面试题很多网络公司都…
考考你:C#常见题型及部分答案
原创:.net读取数据库sql2000
伪静态URL重写配置
配置web.config代码asp.net3.5个性…
使用线程池提高性能 Socket网络编…
ASP.NET(C#)GridView表头的增加…
如何找到正确的学习方向【.NET版】…
  最近更新
C#修改注册表demo
一个获取内容中的图片地址的方法
ASP.NET 4.0尚未在 Web 服务器上注…
四大作用域:application,session…
ConfigurationManager不存在的解决…
vs2012_vs2013_vs2015没有Web Dep…
vs2015禁用解决方案中单击打开文件…
微软为Visual Studio 2015新增安卓…
C#如何实现搜索引擎网络爬虫程序
C#中正则表达式的用法
用C#抓取需要登录的页面数据
VS2015新功能
VS2015安装图解教程
vs2015新功能介绍
vs2015安装图解

关于我们 | 网站建设 | 技术辅导 | 常见问题 | 联系我们 | 友情链接

夜鹰教程网 版权所有 www.yyjcw.com All rights reserved 备案号:蜀ICP备08011740号3