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. hdu2544SPFA板题

    SPFA顾名思义就是更快的最短路算法,是Bellman ford算法的优化,SPFA的平均复杂度大约是O(K*|E|),在一般情况下K大约是小于等于2的数,但是总有人对你心怀不轨,构造一组SPFA最坏 ...

  2. Contest 157

    2019-10-06 12:15:28 总体感受:总体难度一般,dfs和dp题花了点时间,最后一题dp有思路,但是实现上不够好. 注意点:首先是hard问题的覆盖度依然是很大的问题,其次是要注意审题. ...

  3. Mysql性能优化:什么是索引下推?

    导读 索引下推(index condition pushdown )简称ICP,在Mysql5.6的版本上推出,用于优化查询. 在不使用ICP的情况下,在使用非主键索引(又叫普通索引或者二级索引)进行 ...

  4. 收藏 | 15 个你非了解不可的 Linux 特殊字符,妈妈再也不用担心我看不懂这些符号了!

    不知道大家接触 Linux 系统有多久了,可曾了解过 Linux 中有哪些特殊的字符呢?其实啊,那些特殊字符都大有用处呢,今天的文章就给大家简单地科普一下 Linux 中你需要了解的 15 个特殊字符 ...

  5. 家乐的深度学习笔记「4」 - softmax回归

    目录 softmax回归 分类问题 softmax回归模型 softmax运算 矢量表达式 单样本分类的矢量计算表达式 小批量样本分类的矢量计算表达式 交叉熵损失函数 模型预测及评价 图像分类数据集( ...

  6. python逐行读取文本

    一.使用open打开文件后一定要记得调用文件对象的close()方法.比如可以用try/finally语句来确保最后能关闭文件. 二.需要导入import os 三.下面是逐行读取文件内容的三种方法: ...

  7. redis吊锤面试官,这篇足够了!

    原理篇 redis 时单线程的为什么还能那么快? 数据都在内存中,运算都是内存级别的运算. redis既然是单线程的为什么能处理那么多的并发数? 多路复用,操作系统时间轮训epoll 函数作为选择器, ...

  8. Q - 迷宫问题 POJ - 3984(BFS / DFS + 记录路径)

    Q - 迷宫问题 POJ - 3984 定义一个二维数组: int maze[5][5] = { 0, 1, 0, 0, 0, 0, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0, 1, ...

  9. Shell:Day10

    shell脚本:明白一点:shell脚本本身是一个工具 在写shell脚本之前,就要明白:这个功能能到底如何实现? curl 访问文件源代码,查看网站状态: 才能通过shell(bash)所提供的逻辑 ...

  10. 原生JS实现Ajax的跨域请求

    原生JS如何实现Ajax的跨域请求? 在解决这个问题之前,我们务必先清楚为什么我们要跨域请求,以及在什么情况下会跨域请求. 了解一下:“同源策略”,你就知道了: 同源策略限制从一个源加载的文档或脚本如 ...