博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
C#委托与事件
阅读量:6314 次
发布时间:2019-06-22

本文共 6005 字,大约阅读时间需要 20 分钟。

using System;

using System.Collections.Generic;
using System.Linq;
using System.Text;
//****************************************事件与委托*************************************************************
 /* Observer设计模式中主要包括如下两类对象:
Subject:监视对象,它往往包含着其他对象所感兴趣的内容。在本范例中,热水器就是一个监视对象,它包含的其他对象所感兴趣的内容,就是temprature字段,当这个字段的值快到100时,会不断把数据发给监视它的对象。 Observer:监视者,它监视Subject,当Subject中的某件事发生的时候,会告知Observer,而Observer则会采取相应的行动。在本范例中,Observer有警报器和显示器,它们采取的行动分别是发出警报和显示水温。 在本例中,事情发生的顺序应该是这样的:
警报器和显示器告诉热水器,它对它的温度比较感兴趣(注册)。 热水器知道后保留对警报器和显示器的引用。 热水器进行烧水这一动作,当水温超过95度时,通过对警报器和显示器的引用,自动调用警报器的MakeAlert()方法、显示器的ShowMsg()方法。 类似这样的例子是很多的,GOF对它进行了抽象,称为Observer设计模式:Observer设计模式是为了定义对象间的一种一对多的依赖关系,以便于当一个对象的状态改变时,其他依赖于它的对象会被自动告知并更新。Observer模式是一种松耦合的设计模式。* /
/* 委托类型的名称都应该以EventHandler结束。 委托的原型定义:有一个void返回值,并接受两个输入参数:一个Object 类型,一个 EventArgs类型(或继承自EventArgs)。 事件的命名为 委托去掉 EventHandler之后剩余的部分。 继承自EventArgs的类型应该以EventArgs结尾。
 * 再做一下说明:
委托声明原型中的Object类型的参数代表了Subject,也就是被监视对象,在本例中是 Heater(热水器)。回调函数(比如Alarm的MakeAlert)可以通过它访问触发事件的对象(Heater)。 EventArgs 对象包含了Observer所感兴趣的数据,在本例中是temperature。上面这些其实不仅仅是为了编码规范而已,这样也使得程序有更大的灵活性。比如说,如果我们不光想获得热水器的温度,还想在Observer端(警报器或者显示器)方法中获得它的生产日期、型号、价格,那么委托和方法的声明都会变得很麻烦,而如果我们将热水器的引用传给警报器的方法,就可以在方法中直接访问热水器了。*/
namespace DelegateLearning
{
    //热水器
    public class Heater
    {
        private int temperature;
        public string type = "RealFire 001"; // 添加型号作为演示
        public string area = "China Xi'an"; // 添加产地作为演示
        /*声明委托中,sender为被监视对象,回调函数通过它访问触发事件的对象;e为EventArgs对象包含了Observer所感兴趣的数据,即被监视的信息。
        /*委托类型的名称都应该以EventHandler结束。*/
        /*委托是一个类,它定义了方法的类型,使得可以将方法当作另一个方法的参数来进行传递,这种将方法动态地赋给参数的做法,可以避免在程序中大量使用If-Else(Switch)语句,同时使得程序具有更好的可扩展性。*/
        public delegate void BoiledEventHandler(Object sender, BoiledEventArgs e);//声明委托
        /*事件名为委托名除去EventHandler剩下的部分*/
        //声明一个事件类似于声明一个进行了封装的委托类型的变量
        public event BoiledEventHandler Boiled; //声明事件
        // 定义BoiledEventArgs类,传递给Observer所感兴趣的信息
        public class BoiledEventArgs : EventArgs //继承EventArgs
        {
            public readonly int temperature;//temperature为类BoiledEventArgs的只读属性
            public BoiledEventArgs(int temperature)//构造函数,没有返回值,也不用void修饰
            {
                this.temperature = temperature;
            }
        }
        // 可以供继承自 Heater 的类重写,以便继承类拒绝其他对象对它的监视
        protected virtual void OnBoiled(BoiledEventArgs e)
        {
            if (Boiled != null)// 如果有对象注册,事件不为空
            {
                Boiled(this, e); // 调用所有注册对象的方法
            }
        }
        // 烧水
        public void BoilWater()
        {
            for (int i = 0; i <= 100; i++)
            {
                temperature = i;
                if (temperature > 95)
                {
                    //建立BoiledEventArgs 对象。
                    BoiledEventArgs e = new BoiledEventArgs(temperature);
                    OnBoiled(e); // 调用 OnBoiled方法
                }
            }
        }
    }
    // 警报器
    public class Alarm
    {
        public void MakeAlert(Object sender, Heater.BoiledEventArgs e)
        {
            Heater heater = (Heater)sender; //这里是不是很熟悉呢?
            //访问 sender 中的公共字段
            Console.WriteLine("Alarm:{0} - {1}: ", heater.area, heater.type);
            Console.WriteLine("Alarm: 嘀嘀嘀,水已经 {0} 度了:", e.temperature);
            Console.WriteLine();
        }
    }
    // 显示器
    public class Display
    {
        public static void ShowMsg(Object sender, Heater.BoiledEventArgs e)
        { //静态方法
            Heater heater = (Heater)sender;
            Console.WriteLine("Display:{0} - {1}: ", heater.area, heater.type);
            Console.WriteLine("Display:水快烧开了,当前温度:{0}度。", e.temperature);
            Console.WriteLine();
        }
    }
    class Program
    {
        static void Main()
        {
            Heater heater = new Heater();
            Alarm alarm = new Alarm();
            //以下给对象heater中的事件Boiled注册方法
            //将方法封装(注册)在委托类型变量(事件)里
            heater.Boiled += alarm.MakeAlert; //注册方法
            heater.Boiled += (new Alarm()).MakeAlert; //给匿名对象注册方法
            heater.Boiled += new Heater.BoiledEventHandler(alarm.MakeAlert); //也可以这么注册
            heater.Boiled += Display.ShowMsg; //注册静态方法
            heater.BoilWater(); //烧水,会自动调用注册过对象的方法
        }
    }
}
/*
 * 理解:事件的响应方法其实就是事件所封装的方法本身,如方法private void button1_Click(object sender, System.EventArgs e) 就是对事件button1_Click的响应方法,可以通过+=给事件注册其它方法。
 * private void OtherSomething(object sender,System.EventArgs e)
 * this.button1.Click += new System.EventHandler(OtherSomething);
 * 使用委托可以将多个方法绑定到同一个委托变量(事件),当调用此变量时(这里用“调用”这个词,是因为此变量代表一个方法),可以依次调用所有绑定的方法。
 *
 * Event封装了委托类型的变量。在类的内部,不管声明它是public还是protected,它总是private的。在类的外部,注册“+=”和注销“-=”的访问限定符与在声明事件时使用的访问符相同。
 * ********************************************( object sender , EventArgs e )*********************************
 * ( object sender , EventArgs e ) 是C#里面的事件响应的代码
EventArgs是包含事件数据的类的基类,用于传递事件的细节。
EventHandler是一个委托(其在.Net类库中是如下声明的)
委托声明为:public delegate void EventHandler( object sender , EventArgs e )
所以,所有形如:  void 函数名(object 参数名,EventArgs 参数名); 的函数都可以作为Control类的Click事件响应方法了。Object的参数名一般用Source或Sender来表示,两个没有区别。
如下面所定义的一个事件响应方法:
private void button1_Click(object sender, System.EventArgs e)
参数object sender表示引发事件的对象(其实这里传递的是对象的引用,如果是button1的click事件则sender就是button1)
* System.EventArgs e 代表事件的相应信息。
下面我们可以看下Button类的事件声明,以Click事件为例。
public event EventHandler Click;
这里定义了一个EventHandler类型的事件Click
private void button1_Click(object sender, System.EventArgs e)
             {
                       ...
                 }
这是我们和button1_click事件所对应的方法。
那我们怎么把这个方法和事件联系起来呢,请看下面的代码。
this.button1.Click += new System.EventHandler(this.button1_Click);
 * 当button1按钮触发点击事件时再次触发this.button1_Click方法(+= 在原有基础上再加上,就是重新添加)
把this.button1_Click方法绑定到this.button1.Click事件。
 *
以上原理简单理解下就可以了,在实际操作中我们只需要在代码里面调用Web控件里面使用button控件,里面的属性OnClick="button1_Click" 语句便可以起到调用方法的功能了。在VS.NET中可以直接在设计页面加入button, 而上面红色的那行代码编译器会自动实现(可在cs代码文件里面看到)。
 *
 * *****************************************************EventHandler*************************************************************
 * EventHandler表示将处理不包含事件数据的事件的方法。
命名空间:System
程序集:mscorlib(在 mscorlib.dll 中)
C#
[SerializableAttribute]
[ComVisibleAttribute(true)]
public delegate void EventHandler (Object sender,EventArgs e)
 * sender 事件源, e 不包含任何事件数据的 EventArgs。
备注:
.NET Framework 中的事件模型基于具有事件委托,该委托将事件与事件处理程序连接。引发事件需要两个元素:
一个是标识对事件提供响应的方法的委托,另一个是保存事件数据的类。
委托是一个定义签名的类型,即方法的返回值类型和参数列表类型。可以使用委托类型来声明一个变量,该变量可以引用与委托签名相同的所有方法。
事件处理程序委托的标准签名定义一个没有返回值的方法,其第一个参数的类型为 Object,它引用引发事件的实例,第二个参数从 EventArgs 类型派生,它保存事件数据。如果事件不生成事件数据,则第二个参数只是 EventArgs 的一个实例。否则,第二个参数为从 EventArgs 派生的自定义类型,提供保存事件数据所需的全部字段或属性。
EventHandler 是一个预定义的委托,专用于表示不生成数据的事件的事件处理程序方法。如果事件生成数据,则必须提供自己的自定义事件数据类型,并且必须要么创建一个委托,其中第二个参数的类型为自定义类型,要么使用泛型 EventHandler 委托类并用自定义类型替代泛型类型参数。
若要将事件与处理事件的方法关联,请向事件添加委托的实例。除非移除了该委托,否则每当发生该事件时就调用事件处理程序。
 */

转载于:https://www.cnblogs.com/getyoulove/p/3656096.html

你可能感兴趣的文章
segment
查看>>
获取鼠标的原始移动值
查看>>
Linux信号 编程
查看>>
有关滚动与位置
查看>>
Box2D自定义重力
查看>>
chpasswd
查看>>
mysqldump --single-transaction 和--lock-tables参数详解
查看>>
android 数据库_sql语句总结
查看>>
python购物车
查看>>
解决python2和python3的pip冲突
查看>>
面试/编程
查看>>
linux每日命令(16):head命令
查看>>
公司内部分享【富有成效的每日站会】总结
查看>>
打造一个上传图片到图床利器的插件(Mac版 开源)
查看>>
iOS横竖屏
查看>>
thinkphp判断更新是否成功
查看>>
Do While ... Loop 与 Do Until ... Loop 的区别
查看>>
【Linux】查询某个字符串出现次数
查看>>
高效使用jquery之一:请使用'On'函数
查看>>
冲刺第一周第三天
查看>>