事件:定义了事件成员的类允许通知其他其他对象发生了特定的事情。具体的说,定义了事件成员的类能提供以下功能
1.方法能登记它对事件的关注
2.方法能注销它对事件的关注
3.事件发生时,登记了的方法将收到通知
类型之所以能提供事件通知功能,是因为类型维护了一个已登记方法的列表。事件发生后,类型将通知列表中所有已登记的方法。
事件是以委托为基础。委托是调用回调方法的一种类型安全的方式。对象凭借回调方法接收他们订阅的通知。
假如有一下场景:要设计一个电子邮件程序。当有新的邮件的到达时,用户希望做些一别的操作,例如转发给其他人或其他想实现的功能。事件在其中起到的就是一个通知的作用,告诉其他对象有新的邮件到达了,可以做XXX事情了。
下面使用事件实现这个功能
1.定义一个附加信息类,用来通知接收者发生了什么。
/// <summary> /// 事件附加消息 /// </summary> public class NewMailEventArgs:EventArgs{ private readonly string m_from,m_to,m_subject; public NewMailEventArgs(string from,string to,string subject){ m_from=from; m_to=to; m_subject=subject; } // 发件人 public string From { get{return m_from;} } // 接收人 public string To { get{return m_to;} } // 主题 public string Subject{get{return m_subject;}} }
附加信息类继承了EventArgs,这个基类只定义了一个空的信息,在没有附加信息时可直接使用EventArgs.Empty。EventArgs类的源代码
namespace System { // // Summary: // Represents the base class for classes that contain event data, and provides a // value to use for events that do not include event data. public class EventArgs { // // Summary: // Provides a value to use with events that do not have event data. public static readonly EventArgs Empty; // // Summary: // Initializes a new instance of the System.EventArgs class. public EventArgs(); } }
2.定义事件成员
事件成员使用C#关键字event定义。每个事件成员都要指定以下内容:可访问标识符public(因为只有publi才能使其他对象访问),委托类型以及名称。
public class MailManager{ // 定义事件成员 public event EventHandler<NewMailEventArgs> NewMail; }
它的类型是EventHandler<NewMailEventArgs> 这意味着事件通知的所有接收者都必须有一个和这个类型匹配的回调方法。System.EventHandler的委托定义类型如下:
namespace System { // // Summary: // Represents the method that will handle an event when the event provides data. // // Parameters: // sender: // The source of the event. // // e: // An object that contains the event data. // // Type parameters: // TEventArgs: // The type of the event data generated by the event. public delegate void EventHandler<TEventArgs>(object sender, TEventArgs e); }
所以接收者必须提供的方法必须是一下形式:
void MethodName(Object sender,NewMailEventArgs e);
3. 定义负责引发事件的方法来通知事件的登记对象
public class MailManager{ // 定义事件成员 public event EventHandler<NewMailEventArgs> NewMail; // 定义负责引发事件的方法来通知已登记的对象 protected virtual void OnNewMail(NewMailEventArgs e){ // 将字段复制到一个临时变量,避免多线程情况中这个成员被移除 EventHandler<NewMailEventArgs> temp=Volatile.Read(ref NewMail); if(temp!=null) temp(this,e); } // 接受附加信息并调用引发事件的方法来通知所有登记的对象 public void SimulateNewMail(string from,string to,string subject){ NewMailEventArgs e=new NewMailEventArgs(from,to,subject); OnNewMail(e); } }
4. 定义事件接收者
public class Fax{ public Fax(MailManager mm){ // 构造委托实例,向事件登记回调方法 mm.NewMail+=FaxMsg; } /// <summary> /// 回调方法 /// </summary> /// <param name="sender">表示MailManager对象,便于将信息传递给他</param> /// <param name="e">表示MailManager对象想传给我们的附加信息</param> private void FaxMsg(object sender,NewMailEventArgs e){ Console.WriteLine("msg:{0},{1},{2}",e.From,e.To,e.Subject); } /// <summary> /// 注销对事件的登记 /// </summary> /// <param name="mm"></param> public void Unregister(MailManager mm){ mm.NewMail-=FaxMsg; } }
对象不在接收事件通知时应注销对事件的关注。因为对象只要向事件等急了它的一个方法,便不能被垃圾回收。
5. 程序初始化时应首先构造MailManager对象,将指向它的变量传递给Fax。在Fax构造器中添加对事件的关注。最后调用MailManager对象的事件通知方法
static void Main(string[] args) { MailManager mm=new MailManager(); Fax f=new Fax(mm); mm.SimulateNewMail("a","b","Hello World!"); Console.ReadKey(); }
控制台输出结果:以调用回调方法。
以上就是详解C#之事件的详细内容,更多关于C#之事件的资料请关注小牛知识库其它相关文章!
本文向大家介绍C#事件实例详解,包括了C#事件实例详解的使用技巧和注意事项,需要的朋友参考一下 C#事件实例详解 C#和JAVA有许多相似的地方,设计思想差不多,语法及其相像,均传承自面向对象设计思想,灵感来自C++并取其精华去其“糟粕(二字持保留意见)”,中间语言、解释执行、一次编译、到处执行,出身豪门算得上是表兄弟关系,各自拥有庞大的拥趸,两种语言在发展的道路上你追我赶相互借鉴相互学习相互渗透
本文向大家介绍c#中SqlTransaction——事务详解,包括了c#中SqlTransaction——事务详解的使用技巧和注意事项,需要的朋友参考一下 事务处理基本原理 事务是将一系列操作作为一个单元执行,要么成功,要么失败,回滚到最初状态。在事务处理术语中,事务要么提交,要么中止。若要提交事务,所有参与者都必须保证对数据的任何更改是永久的。不论系统崩溃或是发生其他无法预料的事件,更
本文向大家介绍详解C#中委托,事件与回调函数讲解,包括了详解C#中委托,事件与回调函数讲解的使用技巧和注意事项,需要的朋友参考一下 .Net编程中最经常用的元素,事件必然是其中之一。无论在ASP.NET还是WINFrom开发中,窗体加载(Load),绘制(Paint),初始化(Init)等等。 “protected void Page_Load(object sender, EventArgs e
本文向大家介绍C++11 并发指南之Lock 详解,包括了C++11 并发指南之Lock 详解的使用技巧和注意事项,需要的朋友参考一下 在 《 C++11 并发指南三(std::mutex 详解) 》一文中我们主要介绍了 C++11 标准中的互斥量(Mutex),并简单介绍了一下两种锁类型。本节将详细介绍一下 C++11 标准的锁类型。 C++11 标准为我们提供了两种基本的锁类型,分别如下: s
本文向大家介绍Vue.js组件通信之自定义事件详解,包括了Vue.js组件通信之自定义事件详解的使用技巧和注意事项,需要的朋友参考一下 组件通信 从父组件向子组件通信,通过props传递数据就可以了,但Vue组件通信的场景不止有这一种,归纳起来,组件之间的通信可以用下图来表示: 自定义事件 当子组件需要向父组件传递数据时,就要用到自定义事件。子组件用**$ emit()来触发事件**,父组件用**
本文向大家介绍移动web开发之touch事件实例详解,包括了移动web开发之touch事件实例详解的使用技巧和注意事项,需要的朋友参考一下 前面的话 iOS版Safari为了向开发人员传达一些特殊信息,新增了一些专有事件。因为iOS设备既没有鼠标也没有键盘,所以在为移动Safari开发交互性网页时,常规的鼠标和键盘事件根本不够用。随着Android 中的WebKit的加入,很多这样的专有事件变成了