谈一谈C#的事件

C#中事件基于委托,要理解事件要先理解委托,如果觉得自己关于委托不是很了解可以看看我前面写委托的文章

事件基于委托,是一种功能受限的委托,为委托提供了一种发布/订阅机制

使用委托时,一般会出现两种角色:广播者(发布者)和订阅者,这是一个非常常见的模型

依然是用一个非常典型的例子来说明事件,举一个具体的例子就是微信公众号,广播者就是公众号管理员,订阅者就是普通用户。

using System;

class Program
{
static void Main(string[] args)
{
公众号 actionNet = new 公众号();
用户 m = new 用户(); // 订阅事件
actionNet.Published += m.ReceiveMessage; // 执行动作,动作中包含广播行为
actionNet.Publish();
}
} // 定义“发布”委托
public delegate void PublishHandler(); // 特别提醒:为了便于理解,我使用了中文类名,在真实项目中,请勿使用中文关键字
class 公众号
{
public event PublishHandler Published; public void Publish()
{
Console.WriteLine("我是ActionNet,我发布了一条消息"); // 使用了null判断运算符,如果Published为空则不执行后面相关调用
// Published.Invoke()与Published()相同,都是执行委托/事件,后者是前者的简写
// 这条语句就是广播行为
Published?.Invoke();
}
} class 用户
{
public void ReceiveMessage()
{
Console.WriteLine("有一条新的公众号消息到达,请查阅!!!");
}
}

首先广播者,这里的广播者是公众号类,首先定义了一个事件,然后定义了一个方法,在这个方法中触发事件,广播给所有订阅者

订阅者是用户类,只有一个方法

回到Main函数,首先实例化了一个公众号对象和用户对象

然后使用+=将订阅者的方法注册到事件中(事件只能使用+=和-=注册和撤销订阅,后面会讲)

最后是广播者调用了触发事件的方法,向所有订阅者广播(由事件委托执行订阅者注册到事件中的方法)

标准事件模式

.Net Framwork为事件编程定义了一个标准模式,目的是保持框架和用户代码的一致性,核心是System.EventArgs类,但是在.Net Core中并不再要求System.EventArgs类,这个下一节讲

在.Net Framework的标准事件模式中,需要

  • 一个类,继承自EventArgs,类名根据包含的信息来命名,以EventArgs结尾

  • 事件所需的委托(框架定义了名为EventHandler<T>的泛型委托,满足下面的要求,且可以接受所有类型)

    • 委托必须以void作为返回值
    • 委托必须接受两个参数,第一个参数是object类型,第二个参数则是EventArgs的子类。第一个参数表明了事件的广播者,第二个参数则包含了需要传递的额外信息
    • 委托的名称必须以EventHandler结尾
  • 一个protected的虚方法来触发事件,方法名必须和事件名称一致,以On为前缀,并接受唯一的EventArgs参数

下面我们来看前面的例子改写成标准模式的样子

// 带参数的标准模式
using System; class Program
{
static void Main(string[] args)
{
公众号 actionNet = new 公众号();
用户 m = new 用户(); // 订阅事件
actionNet.Published += m.ReceiveMessage; // 执行动作,动作中包含广播行为
actionNet.Publish("人生若只如初见");
}
} // 定义需要传递的额外信息类
class MessageEventArgs : EventArgs
{
public MessageEventArgs(string message)
{
Message = message;
}
public string Message { get; set; }
} // 定义事件的委托,这个委托.Net Framework定义好了的,其实不必再定义,在后面迁移到.Net Core时也改动更少
// public delegate void EventHandler<TEventArgs>(object sender, TEventArgs e) where TEventArgs : EventArgs; // 特别提醒:为了便于理解,我使用了中文类名,在真实项目中,请勿使用中文关键字
class 公众号
{
public event EventHandler<MessageEventArgs> Published; protected virtual void OnPublished(MessageEventArgs e)
{
Console.WriteLine("我是ActionNet,我发布了一条消息"); // this表明事件的广播者是这个类,e是需要传递的额外信息
Published?.Invoke(this,e);
} public void Publish(string content)
{
OnPublished(new MessageEventArgs(content));
}
} class 用户
{
public void ReceiveMessage(object sender,MessageEventArgs e)
{
Console.WriteLine("有一条新的公众号消息到达,请查阅!!!");
Console.WriteLine($"消息来自:{sender}");
Console.WriteLine(e.Message);
}
}
// 不需要额外信息的标准模式
using System; class Program
{
static void Main(string[] args)
{
公众号 actionNet = new 公众号();
用户 m = new 用户(); // 订阅事件
actionNet.Published += m.ReceiveMessage; // 执行动作,动作中包含广播行为
actionNet.Publish("人生若只如初见");
}
} // 不需要用以传递信息的额外信息类 // 定义事件的委托,这个委托.Net Framework定义好了的,其实不必再定义,在后面迁移到.Net Core时也改动更少
// public delegate void EventHandler<TEventArgs>(object sender, TEventArgs e) where TEventArgs : EventArgs; // 特别提醒:为了便于理解,我使用了中文类名,在真实项目中,请勿使用中文关键字
class 公众号
{
// 使用非泛型的EventHandler委托类型
public event EventHandler Published; // 触发事件的方法,直接接收EventArgs类型参数
protected virtual void OnPublished(EventArgs e)
{
Console.WriteLine("我是ActionNet,我发布了一条消息"); // this表明事件的广播者是这个类,e是需要传递的额外信息
Published?.Invoke(this,e);
} public void Publish(string content)
{
//无需额外的信息,则无需创建EventArgs实例,直接返回空
OnPublished(EventArgs.Empty);
}
} class 用户
{
public void ReceiveMessage(object sender,EventArgs e)
{
Console.WriteLine("有一条新的公众号消息到达,请查阅!!!");
Console.WriteLine($"消息来自:{sender}");
}
}

.Net Core的标准事件模式

.NET Core 的标准事件模式较为宽松。 在此版本中,EventHandler<TEventArgs> 定义不再要求 TEventArgs 必须是派生自 System.EventArgs 的类

这个可以看一下官方文档,后面有机会我再更新

事件相对于委托的限制

事件是一种功能受限的委托,具体受限的地方是事件只能使用+=-=,相比于委托少了=(这是其中之一,还理解得不是很透彻,后面再补)

拿前面的代码actionNet.Published += m.ReceiveMessage;这一句是添加订阅,如果不需要了可以使用-=删除订阅,但是不能使用actionNet.Published = null,这样的代码更加健壮

谈一谈C#的事件的更多相关文章

  1. 谈一谈iOS事件的产生和传递

    谈一谈iOS事件的产生和传递 1.事件的产生 发生触摸事件后,系统会将该事件加入到一个由UIApplication管理的事件队列中. UIApplication会从事件队列中取出最前面的事件,并将事件 ...

  2. 谈一谈Java8的函数式编程(二) --Java8中的流

    流与集合    众所周知,日常开发与操作中涉及到集合的操作相当频繁,而java中对于集合的操作又是相当麻烦.这里你可能就有疑问了,我感觉平常开发的时候操作集合时不麻烦呀?那下面我们从一个例子说起. 计 ...

  3. 谈一谈泛型(Generic)

    谈一谈泛型 首先,泛型是C#2出现的.这也是C#2一个重要的新特性.泛型的好处之一就是在编译时执行更多的检查. 泛型类型和类型参数 ​ 泛型的两种形式:泛型类型( 包括类.接口.委托和结构 没有泛型枚 ...

  4. 从一张图开始,谈一谈.NET Core和前后端技术的演进之路

    从一张图开始,谈一谈.NET Core和前后端技术的演进之路 邹溪源,李文强,来自长沙.NET技术社区 一张图 2019年3月10日,在长沙.NET 技术社区组织的技术沙龙<.NET Core和 ...

  5. 谈一谈Elasticsearch的集群部署

      Elasticsearch天生就支持分布式部署,通过集群部署可以提高系统的可用性.本文重点谈一谈Elasticsearch的集群节点相关问题,搞清楚这些是进行Elasticsearch集群部署和拓 ...

  6. 谈一谈对MySQL InnoDB的认识及数据库事物处理的隔离级别

    介绍: InnoDB引擎是MySQL数据库的一个重要的存储引擎,和其他存储引擎相比,InnoDB引擎的优点是支持兼容ACID的事务(类似于PostgreSQL),以及参数完整性(有外键)等.现在Inn ...

  7. 谈一谈APP版本号问题

    如题:谈一谈APP版本号问题 为什么要谈这个问题,周五晚上11~12点,被微信点名,说APP有错,无效的版本号,商城无法下单.我正在准备收拾东西,周末回老家,结果看到这样问题,菊花一紧.我擦,我刚加的 ...

  8. 谈一谈深度学习之semantic Segmentation

    上一次发博客已经是9月份的事了....这段时间公司的事实在是多,有写博客的时间都拿去看paper了..正好春节回来写点东西,也正好对这段时间做一个总结. 首先当然还是好好说点这段时间的主要工作:语义分 ...

  9. 蓝的成长记——追逐DBA(5):不谈技术谈业务,恼人的应用系统

    ***************************************声明*************************************** 个人在oracle路上的成长记录,当中 ...

随机推荐

  1. HDU - 3282 优先队列的使用

    题意: 按照顺序给你n个数,当数的数量是奇数的时候就输出它们的中位数 题解: 优先队列默认是大顶堆,即priority_queue.top()是这个队列中的最大值 那么我们就可以先创造一个大顶堆优先队 ...

  2. [CodeForces-629A 用阶乘会爆掉

    题意: 给你一个n*n的蛋糕,如果某个位置是'C'那就代表这是一个巧克力块,否则就不是.如果某两个巧克力块在同一行或同一列,那么这个家庭的幸福值就会加1,问你这个家庭的幸福值最大是多少 Input 3 ...

  3. 加密算法——RSA算法(c++简单实现)

    RSA算法原理转自:https://www.cnblogs.com/idreamo/p/9411265.html C++代码实现部分为本文新加 RSA算法简介 RSA是最流行的非对称加密算法之一.也被 ...

  4. python try异常处理

    什么是异常 python异常捕获,在刚开始学的时候,经常会遇到两种报错信息:语法错误和执行的异常. 语法错误在执行的时候就会报错,同时控制端会告诉你错误所在的行: 但即便python程序语法是正确的, ...

  5. 1.nginx安装和基本配置

    作者 微信:tangy8080 电子邮箱:914661180@qq.com 更新时间:2019-07-10 20:56:10 星期三 欢迎您订阅和分享我的订阅号,订阅号内会不定期分享一些我自己学习过程 ...

  6. 2.安装Helm

    作者 微信:tangy8080 电子邮箱:914661180@qq.com 更新时间:2019-06-25 13:54:15 星期二 欢迎您订阅和分享我的订阅号,订阅号内会不定期分享一些我自己学习过程 ...

  7. Python 相关

    循环使用迭代器 Python中使用for循环时,内置容器的默认循环都是有迭代器的.使用迭代器每次只会读出一小部分到内存,不断往后next. 通常建议在没有必要的情况下不使用for key,value ...

  8. redis运维与开发笔记

  9. 2019牛客多校第二场F Partition problem(暴搜)题解

    题意:把2n个人分成相同两组,分完之后的价值是val(i, j),其中i属于组1, j属于组2,已知val表,n <= 14 思路:直接dfs暴力分组,新加的价值为当前新加的人与不同组所有人的价 ...

  10. C++ 0LL

    C++ 0LL C plus plus L / l means long LL === long long int countDigitOne(int n) { int countr = 0; for ...