C#委托和事件

Wesley13
• 阅读 720

0. 前言

事件和委托是C#中的高级特性,也是C#中很有意思的一部分。出现事件的地方,必然有委托出现;而委托则不一定会有事件出现。那为什么会出现这样的关系呢?这就需要从事件和委托的定义出发,了解其中的内在。

1. 委托

说起委托,就不得不回忆一下之前在Linq篇中介绍的匿名方法,其中提到了Func和Action这两个类型。这两个类型就是委托。

委托在C#中定义为一种面向对象形式的方法寻址方案。简单来讲,就是定义一个类型,然后表示这个类型代表某一种方法。而委托对象,就是方法参数化。委托可以实现将方法当做一个参数传递给另一个方法,也可以认为是反射中的MethodInfo的一种特例(实际上并没有太多关系)。

委托不关心方法叫什么,也不关心方法从哪来(归属于哪个类或者哪个对象),只关心方法需要哪些参数,返回什么类型。

说到这里,我们来看一下如何定义一个委托吧,委托的定义形式如下:

delegate <返回类型>  委托名(参数列表);//参数列表代表任意个参数复制代码

由之前的定义形式,我们可以知道委托也是一种类型,所以它的定义也符合类型的定义规范。现在我们定义一个没有返回值也没有参数类型的委托作为我们创建的第一个委托:

public delegate void FirstDel();// 类型名称是 FirstDel复制代码

简单的使用一下:

FirstDel del ;
del();// 会直接报错复制代码

上述代码如果运行的话,会很直接的报错,因为你没有告诉编译器变量del 应该是什么,也就是没有为del赋值,同时委托可以赋值为null,所以在使用的时候需要注意不能为null,否者也是无法运行的。

这里应用匿名方法的话,可以按照下面的代码对del进行赋值:

del = ()=>
{
     //省略方法   
}复制代码

那么我们热身结束,开始正式创建一个有意义的委托:

public delegate decimal CalculateArea(decimal height, decimal weight);复制代码

上述委托声明了一个计算面积的规范,使用长宽进行面积计算,那么我们来为它赋值:

CalculateArea squrare = (height, weight) => height * height;// 正方形
CalculateArea rectangle = (height, weight) => height * weight;// 矩形
CalculateArea triangle = (height, weight) => height * weight / 2; //三角形复制代码

我们依次创建了三个计算面积的方法,分别是正方形、矩形、三角形,分别调用它们将会得到对应的计算结果:

var squrareArea = squrare(10, 10);// 100
var rectangleArea = rectangle(19, 10);//190
var triangleArea = triangle(10, 5);//25复制代码

特别的,C#中委托支持多路广播,所以也可以使用+-进行注册和删除。多路广播是指在事件和委托中有多个监听器或响应方法,当事件触发或者委托调用的时候,注册的方法组将会都调用。当使用这种方式对委托进行赋值的时候,委托将自动转为方法组,简单理解就是 委托对象内部创建了一个列表,然后把赋值给它的方法都存进去了。

所以就会产生如下操作:

CalculateArea calculate = squrare;// calculate必须先赋值一个方法
calculate += rectangle;// 增加 矩形的面积计算方法
calculate += triangle; // 增加三角形的面积计算方法
calculate -= triangle; // 减去三角形的面积计算方法复制代码

到这里会产生一个疑问,calculate运行结果是什么,会返回一个数组或者其他类型吗?显然不会,因为calculate定义的返回类型就是一个decimal,所以不会返回其他的值。

嗯,这就产生了另一个疑问,返回的是哪一个方法的计算结果呢,其他方法的计算结果呢?这里告诉大家一个结果,只会返回最后一次注册的方法的执行结果,其他的方法执行了,但是方法的执行结果无法用变量接到。

所以这里有一个很重要的实践,如果有需要把委托当做一个方法列表进行使用的时候,最好声明为void或者抛弃返回值的具体内容。

2. 事件

事件,event。在C#中,事件就像是一种机制,在程序运行到一定阶段的时候或者遇到某些状况的时候,就会触发一个事件。然后如果有其他代码订阅了这个事件,就会自动执行订阅的代码。描述起来很抽象,简单来讲就是在类声明一个委托,并标记这个委托是一个事件,在另一个方法中执行这个事件。其中,触发这个事件的类称为发布者,接受或者注册了处理方法的类称为订阅者。

如何创建或声明一个事件?声明一个事件有两种方式,一种是直接使用EventHandler ,另一种是自己先定义一个委托,然后用这个委托定义事件。

1. 使用EventHandler

public class EventDemo
{
    public event EventHandler HandlerEvent;
}复制代码

2. 使用自定义委托

public class EventDemo
{
    public delegate void EventDelegate(object sender, EventArgs e);
    public event EventDelegate DelegateEvent;
}复制代码

一般事件的定义约定俗称是一个void方法,第一个参数是sender表示事件的发布者,默认是object类型,第二个参数是EventArgs类型的事件变量,表示触发事件时需要订阅者注意的内容,一般用来传一些参数。

其中 EventHandler有一个泛型版本,其声明如下:

public delegate void EventHandler<TEventArgs>(object sender, TEventArgs e);复制代码

其第二个参数并没有对TEventArgs进行限制,所以我们可以用任何类型当做事件变量。

我们再来看看,EventArgs里有什么,什么都没有,只有一个默认构造方法和几个继承自Object的方法。所以在开发中,我们会自己定义一个事件变量类型,为了保持一致会继承EventArgs。

C#建议事件的定义以On开头,表示在什么时触发,示例代码并不符合这个规范。

3. 使用一下事件和委托

创建一个带事件的类:

public class EventDemo
{
    public delegate void EventDelegate(object sender, EventArgs e);

    public event EventDelegate DelegateEvent;


    public void Trigger()
    {
        if (DelegateEvent != null)// 触发事件,按需判断事件的订阅者列表是否为空
        {
            DelegateEvent(this, new EventArgs());
        }
    }
}复制代码

使用一下:

EventDemo demo = new EventDemo(); 
demo.DelegateEvent += (sender, eventArgs) =>
{
    //省略订阅者的方法内容
}
demo.Trigger();//触发事件          复制代码

当发布者尝试触发事件的时候,订阅者将会接收到消息,然后注册订阅者方法就会被调用。发布者向订阅者传递一对sender和eventArgs,订阅者按照自己的逻辑进行处理。

这里很明显可以看出,事件的处理程序注册方法用的+=,所以与之对应的也有一个-=表示取消订阅。

到这里,委托和事件的基本概念就已经介绍完毕了,当然还是那句话,更多的内容在实践中。C#的事件机制让程序员有更多的自由去自定义事件,而不是被局限在某些框架内。所以大家可以多试试C#的事件,也许能发现更多的我不知道的内容呢。

点赞
收藏
评论区
推荐文章
Dax Dax
3年前
jQuery 的事件绑定和事件委托(事件代理)
简单以jQuery的on()方法为例说明:API:1.on(events,selector,data,handler) 如果on()方法的selector参数为空,事件处理程序就被称为直接绑定。每当在被绑定元素上(如下例中被绑定的document元素,译者注)发生事件时,无论这个事件发生在这个元素上还是从内层元素经冒泡而
待兔 待兔
4年前
《C# 教程》菜鸟教程学习笔记
C简介下面列出C一些重要的功能:布尔条件(BooleanConditions)自动垃圾回收(AutomaticGarbageCollection)标准库(StandardLibrary)组件版本(AssemblyVersioning)属性(Properties)和事件(Events)委托(D
Wesley13 Wesley13
3年前
C# 1.0 新特性之异步委托(AP、APM)
Ø前言C异步委托也是属于异步编程中的一种,可以称为AsynchronousProgramming(异步编程)或者AsynchronousProgrammingModel(异步编程模型),因为这是实现异步编程的模式。委托是C1.0就有的特性,并且.NETv1.0同时也伴随有AsyncCallback、IAsyncResult
Stella981 Stella981
3年前
C# lambda 和 Linq
本章节给大家带来的是Lambda和Linq的关系Lambda:是实例化委托的一个参数,也就是一个方法Linq:是基于委托(lambda)的封装,代码重用,逻辑解耦,是一个帮助类库,linq是用泛型,委托,lamda实现的,总的来说:把对数据操作的通用部分完成,把可变的交给委托,使用者只关心可变部分。一、Lambda文章的开始
Stella981 Stella981
3年前
Logback
1\.作用Logback将写日志事件的任务委托给appender组件完成,SiftingAppender顾名思义就是筛选日志事件,具体点就是:对于Logback委托给它的日志事件,SiftingAppender会对日志事件做一些区分,然后不同的事件SiftingAppender会委托不同的appender去完成真正的写操作。假
Wesley13 Wesley13
3年前
C# 事件与委托
一些小概念,防止自己太久不用忘记了事件委托:Action<int,int,string….(maxnumberof16parameters)a永远没有返回值,最多16个泛型输入参数e.g.:StaticvoidPrintStr(stringstr){   Console.WriteLine(str);
Wesley13 Wesley13
3年前
C#中委托和事件的区别
大致来说,委托是一个类,该类内部维护着一个字段,指向一个方法。事件可以被看作一个委托类型的变量,通过事件注册、取消多个委托或方法。本篇分别通过委托和事件执行多个方法,从中体会两者的区别。□通过委托执行方法classProgram{staticvoidMain(stringargs){
Wesley13 Wesley13
3年前
Unity事件系统
\1.前言Unity中事件/委托有着广泛的应用,本文通过封装一个简易的事件的系统,来统一管理消息的传递。此功能在简易应用或者事件较少的体现不出太好的作用,但是对于事件应用较多时,可以减少脚本之间的耦合。通过此事件系统架起不同脚本之间的桥梁,对于大量应用事件场景中具有良好的效果。\2.事件系统\2.1事件管理类管理事件的
Wesley13 Wesley13
3年前
IOS 委托和协议区别和联系 (=)
上一片大致说了一下IOS上面委托和协议的区别和联系,并且举了一个简单的例子,但是例子比较简单,今天做一个用委托模拟button回调的例子。在一个自定义View上面放一个登陆按钮,并且这个LoginView里面有一个实现ILogin的委托对象,在登陆按钮的点击事件中调用需要实现的协议函数。在一个ViewController中实现ILgin协议,并实现log
Stella981 Stella981
3年前
JavaScript事件属性event.target和currentTarget 属性的区别。
event.target获取的是触发事件的标签元素event.currentTarget获取到的是发起事件的标签元素一、事件属性:event.targettarget事件委托的定义:本来该自己干的事,但是自己不干,交给别人来干例子1!(https://oscimg.oschina.n