C# 基础知识系列- 11 委托和事件
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 委托和事件的更多相关文章
- [C# 基础知识系列]专题四:事件揭秘
转自http://www.cnblogs.com/zhili/archive/2012/10/27/Event.html 引言: 前面几个专题对委托进行了详细的介绍的,然后我们在编写代码过程中经常会听 ...
- [C# 基础知识系列]专题四:事件揭秘 (转载)
引言: 前面几个专题对委托进行了详细的介绍的,然后我们在编写代码过程中经常会听到“事件”这个概念的,尤其是写UI的时候,当我们点击一个按钮后VS就会自动帮我们生成一些后台的代码,然后我们就只需要在Cl ...
- C# 基础知识系列- 12 任务和多线程
0. 前言 照例一份前言,在介绍任务和多线程之前,先介绍一下异步和同步的概念.我们之间介绍的知识点都是在同步执行,所谓的同步就是一行代码一行代码的执行,就像是我们日常乘坐地铁通过安检通道一样,想象我们 ...
- C# 基础知识系列- 10 反射和泛型(二)
0. 前言 这篇文章延续<C# 基础知识系列- 5 反射和泛型>,继续介绍C#在反射所开发的功能和做的努力.上一篇文章大概介绍了一下泛型和反射的一些基本内容,主要是通过获取对象的类型,然后 ...
- 学习javascript基础知识系列第二节 - this用法
通过一段代码学习javascript基础知识系列 第二节 - this用法 this是面向对象语言中的一个重要概念,在JAVA,C#等大型语言中,this固定指向运行时的当前对象.但是在javascr ...
- C# 基础知识系列- 3 集合数组
简单的介绍一下集合,通俗来讲就是用来保管多个数据的方案.比如说我们是一个公司的仓库管理,公司有一堆货物需要管理,有同类的,有不同类的,总而言之就是很多.很乱.我们对照集合的概念对仓库进行管理的话,那么 ...
- C# 基础知识系列- 9 字符串的更多用法(一)
0. 前言 在前面的文章里简单介绍了一下字符串的相关内容,并没有涉及到更多的相关内容,这一篇将尝试讲解一下在实际开发工作中会遇到的字符串的很多操作. 1. 创建一个字符串 这部分介绍一下如何创建一个字 ...
- C# 基础知识系列-13 常见类库(三)
0. 前言 在<C# 基础知识系列- 13 常见类库(二)>中,我们介绍了一下DateTime和TimeSpan这两个结构体的内容,也就是C#中日期时间的简单操作.本篇将介绍Guid和Nu ...
- 基础知识系列☞C#中→属性和字段的区别
"好吧...准备写个'基础知识系列',算是记录下吧,时时看看,更加加深记忆···" 其实本来准备叫"面试系列"... 字段.属性.你先知道的哪个概念? ***我 ...
随机推荐
- c语言课程设计--图书/音乐管理系统
这个代码因为配置信息的原因不能直接整个拿去用(o゜▽゜)o☆ 这个代码因为配置信息的原因不能直接整个拿去用(o゜▽゜)o☆ 这个代码因为配置信息的原因不能直接整个拿去用(o゜▽゜)o☆ 只能提供一个思 ...
- 【codeforces】Codeforces Round #612 (Div. 2) C. Garland——DP
题目链接 贪心模拟了半天,最后放弃了 题意 给你一串从1−n1-n1−n的序列,其中部分未知(表示为0),补全序列使得相邻数值奇偶性相反的数量最少 相邻数值的奇偶性相反:两个相邻的两个数值,其中一个为 ...
- [gcd,灵感] Codeforces 1200C Round Corridor
题目:https://codeforces.com/contest/1200/problem/C C. Round Corridor time limit per test 1 second memo ...
- window的三种系统弹框介绍
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8 ...
- XCode8 安装模拟器
1.下载模拟器 参考:http://blog.csdn.net/piratest/article/details/52538978 参考:http://blog.csdn.net/zhangao008 ...
- Mac 中命令行启动、停止、重启Mysql
启动: ~$ sudo /usr/local/mysql/support-files/mysql.server start 停止: ~$ sudo /usr/local/mysql/support-f ...
- html前端之基础篇
HTML介绍 Web服务本质 import socket sk = socket.socket() sk.bind(("127.0.0.1", 8080)) sk.listen ...
- Unity 游戏框架搭建 2019 (二十一、二十二) 第三章简介&整理前的准备
整理前的准备 到目前为止,我们积攒了很多示例了,并且每个示例也都贯彻了最的约定和规则. 在上一篇的小结也说了一个比较新的东西:编程体验优化. 在之前我们还积攒了一个问题:代码重复问题. 我们可是忍住整 ...
- docker安装Elasticsearch7.6集群并设置密码
docker安装Elasticsearch7.6集群并设置密码 Elasticsearch从6.8开始, 允许免费用户使用X-Pack的安全功能, 以前安装es都是裸奔.接下来记录配置安全认证的方法. ...
- vue中keepAlive的用法[返回页面不刷新]
本文转载于時間蒼白了誓言_49b9 使用vue单页开发项目时遇到一个问题:在列表页进入详情页,按返回键返回列表页时页面刷新了,用户体验非常差啊!!!查阅了一下相关问题,使用解决这个问题,下面是我的使用 ...