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#的事件,也许能发现更多的我不知道的内容呢。

更多内容烦请关注我的博客

C# 基础知识系列- 11 委托和事件的更多相关文章

  1. [C# 基础知识系列]专题四:事件揭秘

    转自http://www.cnblogs.com/zhili/archive/2012/10/27/Event.html 引言: 前面几个专题对委托进行了详细的介绍的,然后我们在编写代码过程中经常会听 ...

  2. [C# 基础知识系列]专题四:事件揭秘 (转载)

    引言: 前面几个专题对委托进行了详细的介绍的,然后我们在编写代码过程中经常会听到“事件”这个概念的,尤其是写UI的时候,当我们点击一个按钮后VS就会自动帮我们生成一些后台的代码,然后我们就只需要在Cl ...

  3. C# 基础知识系列- 12 任务和多线程

    0. 前言 照例一份前言,在介绍任务和多线程之前,先介绍一下异步和同步的概念.我们之间介绍的知识点都是在同步执行,所谓的同步就是一行代码一行代码的执行,就像是我们日常乘坐地铁通过安检通道一样,想象我们 ...

  4. C# 基础知识系列- 10 反射和泛型(二)

    0. 前言 这篇文章延续<C# 基础知识系列- 5 反射和泛型>,继续介绍C#在反射所开发的功能和做的努力.上一篇文章大概介绍了一下泛型和反射的一些基本内容,主要是通过获取对象的类型,然后 ...

  5. 学习javascript基础知识系列第二节 - this用法

    通过一段代码学习javascript基础知识系列 第二节 - this用法 this是面向对象语言中的一个重要概念,在JAVA,C#等大型语言中,this固定指向运行时的当前对象.但是在javascr ...

  6. C# 基础知识系列- 3 集合数组

    简单的介绍一下集合,通俗来讲就是用来保管多个数据的方案.比如说我们是一个公司的仓库管理,公司有一堆货物需要管理,有同类的,有不同类的,总而言之就是很多.很乱.我们对照集合的概念对仓库进行管理的话,那么 ...

  7. C# 基础知识系列- 9 字符串的更多用法(一)

    0. 前言 在前面的文章里简单介绍了一下字符串的相关内容,并没有涉及到更多的相关内容,这一篇将尝试讲解一下在实际开发工作中会遇到的字符串的很多操作. 1. 创建一个字符串 这部分介绍一下如何创建一个字 ...

  8. C# 基础知识系列-13 常见类库(三)

    0. 前言 在<C# 基础知识系列- 13 常见类库(二)>中,我们介绍了一下DateTime和TimeSpan这两个结构体的内容,也就是C#中日期时间的简单操作.本篇将介绍Guid和Nu ...

  9. 基础知识系列☞C#中→属性和字段的区别

    "好吧...准备写个'基础知识系列',算是记录下吧,时时看看,更加加深记忆···" 其实本来准备叫"面试系列"... 字段.属性.你先知道的哪个概念? ***我 ...

随机推荐

  1. JMeter入门介绍

    目录 概述 下载&安装 实战JMetetr 测试计划简述 准备测试计划 编写测试计划 录制测试脚本 执行性能测试 单机测试 分布式测试 分析测试报告 APDEX 响应时间和吞吐量统计 测试结果 ...

  2. python之目录

    一.python基础 ​ python之字符串str操作方法 ​ python之int (整型) ​ python之bool (布尔值) ​ python之str (字符型) ​ python之ran ...

  3. 高数解题神器:拍照上传就出答案,这个中国学霸做的AI厉害了 | Demo

    一位叫Roger的中国学霸小哥的拍照做题程序mathAI一下子火了,这个AI,堪称数学解题神器. 输入一张包含手写数学题的图片,AI就能识别出输入的数学公式,然后给出计算结果. 不仅加减乘除基本运算, ...

  4. CDN 内容分发

    1,传统架构访问服务器资源: www.aiyuesheng.com/page/logo.png 这是部署在服务器上的一张图片,因为服务器部署在上海,所以在上海或周边的人访问要稍微快一点,但是,若是云南 ...

  5. Kubernets中获取客户端真实IP总结

    1. 导言 绝大多数业务场景都是需要知道客户端IP的 在k8s中运行的业务项目,如何获取到客户端真实IP? 本文总结了通行的2种方式 要答案的直接看方式一.方式二和总结 SEO 关键字 nginx i ...

  6. Java复合优先于继承

    复合优于继承 继承打破了封装性(子类依赖父类中特定功能的实现细节) 合理的使用继承的情况: 在包内使用 父类专门为继承为设计,并且有很好的文档说明,存在is-a关系 只有当子类真正是父类的子类型时,才 ...

  7. STM32F103C8T6最小系统开发板原理图

    1.

  8. 自执行函数-[javascript]-[语法]

    在看别人的代码的时候,遇到了一种写法,之前没有见过,如下: ![](https://img2018.cnblogs.com/blog/1735896/201912/1735896-2019122114 ...

  9. 记录一次简单的springboot发送邮件功能

    场景:经常在我们系统中有通过邮件功能找回密码,或者发送生日祝福等功能,今天记录下springboot发送邮件的简单功能 1.引入maven <!-- 邮件开发--><dependen ...

  10. iOS, Xcode11,项目提示第三方库报错无法运行 bundle format unrecognized, invalid, or unsuitable

    检查你有没有把静态库和动态库配置错误!! 下图处是配置动态库的地方! 对于动态库和静态库都有使用的时候,注意把静态库设置成“Do not Embeded”