1.设计要公开事件的类型,该类型用来容纳所有需要发送给事件通知接收者的附加信息。根据约定,该类型应该从System.EventArgs
派生。
例如:
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;
}
...
}
EventArgs类在FCL中的定义:
[ComVisible(true), Serializable]
public class EventArgs{
public static readonly EventArgs Empty = new EventArgs();
public EventArgs(){}
}
可以看出,该类型的实现十分简单,就是一个让其他类型继承的基类型。许多事件都没有附加信息需要传递。例如,当一个Button
向已登记的接收者通知自己被单击时,调用回调方法就可以了。定义不需要传递附加数据的事件时,可以直接使用EventArgs.Empty
,不用构造新的EventArgs
对象。
2.使用关键字event
定义事件成员,使用EventHandler<TEventArgs>
来定义事件成员的类型。
例如:
public event EventHandler<NewMailEventArgs> NewMail;
上述NewMail
是事件名称。
事件成员的类型是EventHandler<NewMailEventArgs>
,其委托定义如下:
public delegate void EventHandler<TEventArgs>(object sender, TEventArgs e);
所以方法原型必须具有以下形式:
void MethodName(object sender, NewMailEventArgs e);
3.定义负责引发事件的方法来通知时间的登记对象,按照约定,该方法应该是一个受保护的虚方法。
protected virtual void OnNewMail(NewMailEventArgs e){
if(NewMail != null) NewMail(this, e);
}
上述方法,有线程安全的问题。在判断NewMail时不为null,但就在调用NewMail之前,另一个线程可能从委托链中移除了一个委托,导致NewMail成了null,所以可以使用以下方法修正这个竞态问题
protected virtual void OnNewMail(NewMailEventArgs e){
// 出于线程安全,将委托字段引用复制到一个临时变量,这里使用Volatile.Read来强迫NewMail在这个条用发生时读取。
var temp = Volatile.Read(ref NewMail);
NewMail?.Invoke(this, e);
}