结合Visual C#开发环境讲解C#中事件的订阅和取消订阅
类或对象可以通过事件向其他类或对象通知发生的相关事情。发送(或引发)事件的类称为“发行者”,接收(或处理)事件的类称为“订户”。
在典型的C#Windows窗体或Web应用程序中,可订阅由控件(如按钮和列表框)引发的事件。可使用VisualC#集成开发环境(IDE)来浏览控件发布的事件,选择要处理的事件。IDE会自动添加空事件处理程序方法和订阅事件的代码。
事件概述
事件具有以下特点:
- 发行者确定何时引发事件,订户确定执行何种操作来响应该事件。
- 一个事件可以有多个订户。一个订户可处理来自多个发行者的多个事件。
- 没有订户的事件永远也不会引发。
- 事件通常用于通知用户操作,例如,图形用户界面中的按钮单击或菜单选择操作。
- 如果一个事件有多个订户,当引发该事件时,会同步调用多个事件处理程序。
在.NETFramework类库中,事件是基于EventHandler委托和EventArgs基类的。
订阅和取消订阅事件
如果您想编写引发事件时调用的自定义代码,则可以订阅由其他类发布的事件。例如,可以订阅某个按钮的click事件,以使应用程序在用户单击该按钮时执行一些有用的操作。
使用VisualStudioIDE订阅事件
如果看不到“属性”窗口,请在“设计”视图中,右击要为其创建事件处理程序的窗体或控件,然后选择“属性”。
在“属性”窗口的顶部,单击“事件”图标。
双击要创建的事件,例如Load事件。
VisualC#会创建一个空事件处理程序方法,并将其添加到您的代码中。或者,您也可以在“代码”视图中手动添加代码。例如,下面的代码行声明了一个在Form类引发Load事件时调用的事件处理程序方法。
privatevoidForm1_Load(objectsender,System.EventArgse) { //Addyourformloadeventhandlingcodehere. }
还会在项目的Form1.Designer.cs文件的InitializeComponent方法中自动生成订阅该事件所需的代码行。该代码行类似于:
this.Load+=newSystem.EventHandler(this.Form1_Load);
以编程方式订阅事件
定义一个事件处理程序方法,其签名与该事件的委托签名匹配。例如,如果事件基于EventHandler委托类型,则下面的代码表示方法存根:
voidHandleCustomEvent(objectsender,CustomEventArgsa) { //Dosomethingusefulhere. }
使用加法赋值运算符(+=)来为事件附加事件处理程序。在下面的示例中,假设名为publisher的对象拥有一个名为RaiseCustomEvent的事件。请注意,订户类需要引用发行者类才能订阅其事件。
publisher.RaiseCustomEvent+=HandleCustomEvent;
请注意,前面的语法是C#2.0中的新语法。此语法完全等效于必须使用new关键字显式创建封装委托的C#1.0语法:
publisher.RaiseCustomEvent+=newCustomEventHandler(HandleCustomEvent);
还可以通过使用lambda表达式添加事件处理程序:
publicForm1() { InitializeComponent(); //Usealambdaexpressiontodefineaneventhandler. this.Click+=(s,e)=>{MessageBox.Show( ((MouseEventArgs)e).Location.ToString());}; }
有关更多信息,请参见如何:在LINQ外部使用Lambda表达式(C#编程指南)。
使用匿名方法订阅事件
如果以后不必取消订阅某个事件,则可以使用加法赋值运算符(+=)将匿名方法附加到此事件。在下面的示例中,假设名为publisher的对象拥有一个名为RaiseCustomEvent的事件,并且还定义了一个CustomEventArgs类以承载某些类型的专用事件信息。请注意,订户类需要引用publisher才能订阅其事件。
publisher.RaiseCustomEvent+=delegate(objecto,CustomEventArgse) { strings=o.ToString()+""+e.ToString(); Console.WriteLine(s); };
请务必注意,如果使用匿名函数订阅事件,事件的取消订阅过程将比较麻烦。这种情况下若要取消订阅,必须返回到该事件的订阅代码,将该匿名方法存储在委托变量中,然后将此委托添加到该事件中。一般来说,如果必须在后面的代码中取消订阅某个事件,则建议您不要使用匿名函数订阅此事件。
取消订阅
要防止在引发事件时调用事件处理程序,请取消订阅该事件。要防止资源泄露,应在释放订户对象之前取消订阅事件。在取消订阅事件之前,在发布对象中作为该事件的基础的多路广播委托会引用封装了订户的事件处理程序的委托。只要发布对象保持该引用,垃圾回收功能就不会删除订户对象。
取消订阅事件
使用减法赋值运算符(-=)取消订阅事件:
publisher.RaiseCustomEvent-=HandleCustomEvent;
所有订户都取消订阅事件后,发行者类中的事件实例将设置为null。