C# 事件的設(shè)計(jì)與使用深入理解
更新時(shí)間:2012年12月24日 16:34:18 作者:
事件是用于通知其他對(duì)象發(fā)生了本對(duì)象發(fā)生了特定的事情的類(lèi)型成員;事件是.NET類(lèi)型成員中相對(duì)較為難以理解和實(shí)踐的一個(gè)成員,因?yàn)槭录亩x不是繼承自基礎(chǔ)的數(shù)據(jù)類(lèi)型,而是對(duì)委托(delegate)的封裝。所以,在了解事件之前,你需要先了解一點(diǎn)委托
相關(guān)概念
定義:事件是用于通知其他對(duì)象發(fā)生了本對(duì)象發(fā)生了特定的事情的類(lèi)型成員。
說(shuō)明:事件是.NET類(lèi)型成員中相對(duì)較為難以理解和實(shí)踐的一個(gè)成員,因?yàn)槭录亩x不是繼承自基礎(chǔ)的數(shù)據(jù)類(lèi)型,而是對(duì)委托(delegate)的封裝。所以,在了解事件之前,你需要先了解一點(diǎn)委托。
應(yīng)用場(chǎng)景:事件的應(yīng)用場(chǎng)景非常廣泛,其中最常見(jiàn)的場(chǎng)景是在各個(gè)前端控件中的大量觸發(fā)事件設(shè)計(jì)。原因是因?yàn)?
意義:事件成員的使用有利于在程序中對(duì)面向?qū)ο笤瓌t的實(shí)現(xiàn)。例如類(lèi)型的單一職責(zé)原則,控制反轉(zhuǎn)原則。設(shè)想如果前端控件不能抽象出大量豐富的事件,那幾乎不能將前端的UI元素與業(yè)務(wù)邏輯脫鉤。程序必然高度耦合。
設(shè)計(jì)模式的應(yīng)用:經(jīng)典設(shè)計(jì)模式中的觀察者模式就非常依賴(lài)于對(duì)事件成員的設(shè)計(jì)而實(shí)現(xiàn)。
本章將通過(guò)設(shè)計(jì)一個(gè)電子郵件到達(dá)時(shí),觸發(fā)事件的場(chǎng)景來(lái)解析對(duì)事件提供者和訂閱者類(lèi)型的設(shè)計(jì)。案例來(lái)源于《CLR Via C#》一書(shū)。
事件提供者類(lèi)型的設(shè)計(jì)
一. 定義類(lèi)型來(lái)容納所有需要發(fā)送給事件訂閱者的附加信息
目標(biāo):定義一個(gè)類(lèi)型用于向事件的訂閱者傳遞信息
方法:繼承默認(rèn)的System.EventArgs類(lèi)型,實(shí)現(xiàn)簡(jiǎn)單的需要傳遞信息的字段,屬性以及實(shí)例構(gòu)造器成員。示例如下:
using System;
using System.Linq;
namespace ConsoleTest
{
public class NewMailEventArgs : EventArgs
{
private readonly string from, to, subject;
public NewMailEventArgs(string from, string to, string subject)
{
this.from = from;
this.to = to;
this.subject = subject;
}
public string Subject
{
get
{
return this.subject;
}
}
public string To
{
get
{
return this.to;
}
}
public string From
{
get
{
return this.from;
}
}
}
}
二. 定義事件成員
目標(biāo):在事件提供者類(lèi)型中定義一個(gè)事件成員,用于事件訂閱者對(duì)象的注冊(cè)。
方法:封裝一個(gè)自定義委托,來(lái)提供事件處理方法的模板;或者實(shí)現(xiàn)一個(gè)System.EventHandler的泛型類(lèi)型來(lái)達(dá)到一樣的效果。(EventHandler是一個(gè)默認(rèn)提供的已封裝的委托)。兩種方法的示例分別如下:
方法一:
public delegate void NewMailHandler(object e, NewMailEventArgs args);
public class MailManager
{
public event NewMailHandler NewMail;
}
方法二:
public class MailManager
{
public event EventHandler<NewMailEventArgs> NewMail;
}
為什么這兩種方法能夠達(dá)到同樣的效果,查看一下System.EventHandler的定義就能知曉:
namespace System
{
// 摘要:
// 表示將處理事件的方法。
//
// 參數(shù):
// sender:
// 事件源。
//
// e:
// 一個(gè)包含事件數(shù)據(jù)的 System.EventArgs。
//
// 類(lèi)型參數(shù):
// TEventArgs:
// 由該事件生成的事件數(shù)據(jù)的類(lèi)型。
[Serializable]
public delegate void EventHandler<TEventArgs>(object sender, TEventArgs e);
}
三. 定義一個(gè)統(tǒng)一觸發(fā)事件的方法入口來(lái)通知事件的訂閱對(duì)象
目標(biāo):在事件提供者類(lèi)型中定義一個(gè)方法成員,用來(lái)統(tǒng)一的引發(fā)目標(biāo)事件。
說(shuō)明:為了保證這個(gè)方法只能在本類(lèi)型及派生類(lèi)型中調(diào)用,我們需要將方法修飾為protected, 為了讓派生類(lèi)型可以重寫(xiě)這個(gè)方法,我們需要將該方法修飾為virtual
意義:這個(gè)統(tǒng)一入口方法的意義在于,能夠統(tǒng)一維護(hù)觸發(fā)事件的方式,并且能夠確保事件調(diào)用的線程安全性。(避免在不同的線程觸發(fā)時(shí),事件訂閱者的狀態(tài)不同步)
示例如下:
public class MailManager
{
public event EventHandler<NewMailEventArgs> NewMail;
protected virtual void OnNewMail(NewMailEventArgs e)
{
//處于線程安全的考慮,現(xiàn)在將對(duì)委托字段的引用復(fù)制到一個(gè)臨時(shí)字段中
EventHandler<NewMailEventArgs> temp = System.Threading.Interlocked.CompareExchange
(ref NewMail, null, null);
//如果有事件訂閱者對(duì)象的存在,則通知他們,事件已觸發(fā)
if (temp != null)
temp(this, e);
}
}
四. 在所有需要觸發(fā)事件的業(yè)務(wù)方法中,調(diào)用第三步中定義的方法
目標(biāo):在類(lèi)型中還需要有一個(gè)業(yè)務(wù)方法,來(lái)將業(yè)務(wù)中的場(chǎng)景轉(zhuǎn)化為事件觸發(fā)。。
方法:在任意需要的業(yè)務(wù)方法中,直接調(diào)用第三步的方法就可以了,不過(guò)需要實(shí)現(xiàn)封裝一個(gè)傳遞信息的類(lèi)型。
示例如下:
public class MailManager
{
public event EventHandler<NewMailEventArgs> NewMail;
protected virtual void OnNewMail(NewMailEventArgs e)
{
//處于線程安全的考慮,現(xiàn)在將對(duì)委托字段的引用復(fù)制到一個(gè)臨時(shí)字段中
EventHandler<NewMailEventArgs> temp = System.Threading.Interlocked.CompareExchange
(ref NewMail, null, null);
//如果有事件訂閱者對(duì)象的存在,則通知他們,事件已觸發(fā)
if (temp != null)
temp(this, e);
}
public void SimulateNewMail(string from, string to, string subject)
{
//構(gòu)造一個(gè)對(duì)象來(lái)封裝向傳給事件訂閱者的信息
NewMailEventArgs e = new NewMailEventArgs(from, to, subject);
//觸發(fā)事件引發(fā)的入口方法
OnNewMail(e);
}
}
事件訂閱者類(lèi)型的設(shè)計(jì)
一. 定義類(lèi)型來(lái)訂閱和偵聽(tīng)事件
目標(biāo):設(shè)計(jì)一個(gè)傳真類(lèi)型Fax類(lèi)來(lái)偵聽(tīng)NewMail事件。
說(shuō)明:Fax類(lèi)型中需要具備對(duì)NewMail事件的訂閱和取消訂閱的方法。示例如下:
internal sealed class Fax
{
private MailManager mailManager;
public Fax(MailManager mm)
{
this.mailManager = mm;
}
public void Register()
{
mailManager.NewMail += new EventHandler<NewMailEventArgs>(FaxMsg);
}
void FaxMsg(object sender, NewMailEventArgs e)
{
Console.WriteLine("Fax mail message");
Console.WriteLine("From = {0}, To = {1}, Subject = {2}", e.From, e.To, e.Subject);
}
public void Unregister()
{
mailManager.NewMail -= FaxMsg;
}
}
定義:事件是用于通知其他對(duì)象發(fā)生了本對(duì)象發(fā)生了特定的事情的類(lèi)型成員。
說(shuō)明:事件是.NET類(lèi)型成員中相對(duì)較為難以理解和實(shí)踐的一個(gè)成員,因?yàn)槭录亩x不是繼承自基礎(chǔ)的數(shù)據(jù)類(lèi)型,而是對(duì)委托(delegate)的封裝。所以,在了解事件之前,你需要先了解一點(diǎn)委托。
應(yīng)用場(chǎng)景:事件的應(yīng)用場(chǎng)景非常廣泛,其中最常見(jiàn)的場(chǎng)景是在各個(gè)前端控件中的大量觸發(fā)事件設(shè)計(jì)。原因是因?yàn)?
意義:事件成員的使用有利于在程序中對(duì)面向?qū)ο笤瓌t的實(shí)現(xiàn)。例如類(lèi)型的單一職責(zé)原則,控制反轉(zhuǎn)原則。設(shè)想如果前端控件不能抽象出大量豐富的事件,那幾乎不能將前端的UI元素與業(yè)務(wù)邏輯脫鉤。程序必然高度耦合。
設(shè)計(jì)模式的應(yīng)用:經(jīng)典設(shè)計(jì)模式中的觀察者模式就非常依賴(lài)于對(duì)事件成員的設(shè)計(jì)而實(shí)現(xiàn)。
本章將通過(guò)設(shè)計(jì)一個(gè)電子郵件到達(dá)時(shí),觸發(fā)事件的場(chǎng)景來(lái)解析對(duì)事件提供者和訂閱者類(lèi)型的設(shè)計(jì)。案例來(lái)源于《CLR Via C#》一書(shū)。
事件提供者類(lèi)型的設(shè)計(jì)
一. 定義類(lèi)型來(lái)容納所有需要發(fā)送給事件訂閱者的附加信息
目標(biāo):定義一個(gè)類(lèi)型用于向事件的訂閱者傳遞信息
方法:繼承默認(rèn)的System.EventArgs類(lèi)型,實(shí)現(xiàn)簡(jiǎn)單的需要傳遞信息的字段,屬性以及實(shí)例構(gòu)造器成員。示例如下:
復(fù)制代碼 代碼如下:
using System;
using System.Linq;
namespace ConsoleTest
{
public class NewMailEventArgs : EventArgs
{
private readonly string from, to, subject;
public NewMailEventArgs(string from, string to, string subject)
{
this.from = from;
this.to = to;
this.subject = subject;
}
public string Subject
{
get
{
return this.subject;
}
}
public string To
{
get
{
return this.to;
}
}
public string From
{
get
{
return this.from;
}
}
}
}
二. 定義事件成員
目標(biāo):在事件提供者類(lèi)型中定義一個(gè)事件成員,用于事件訂閱者對(duì)象的注冊(cè)。
方法:封裝一個(gè)自定義委托,來(lái)提供事件處理方法的模板;或者實(shí)現(xiàn)一個(gè)System.EventHandler的泛型類(lèi)型來(lái)達(dá)到一樣的效果。(EventHandler是一個(gè)默認(rèn)提供的已封裝的委托)。兩種方法的示例分別如下:
方法一:
復(fù)制代碼 代碼如下:
public delegate void NewMailHandler(object e, NewMailEventArgs args);
public class MailManager
{
public event NewMailHandler NewMail;
}
方法二:
復(fù)制代碼 代碼如下:
public class MailManager
{
public event EventHandler<NewMailEventArgs> NewMail;
}
為什么這兩種方法能夠達(dá)到同樣的效果,查看一下System.EventHandler的定義就能知曉:
復(fù)制代碼 代碼如下:
namespace System
{
// 摘要:
// 表示將處理事件的方法。
//
// 參數(shù):
// sender:
// 事件源。
//
// e:
// 一個(gè)包含事件數(shù)據(jù)的 System.EventArgs。
//
// 類(lèi)型參數(shù):
// TEventArgs:
// 由該事件生成的事件數(shù)據(jù)的類(lèi)型。
[Serializable]
public delegate void EventHandler<TEventArgs>(object sender, TEventArgs e);
}
三. 定義一個(gè)統(tǒng)一觸發(fā)事件的方法入口來(lái)通知事件的訂閱對(duì)象
目標(biāo):在事件提供者類(lèi)型中定義一個(gè)方法成員,用來(lái)統(tǒng)一的引發(fā)目標(biāo)事件。
說(shuō)明:為了保證這個(gè)方法只能在本類(lèi)型及派生類(lèi)型中調(diào)用,我們需要將方法修飾為protected, 為了讓派生類(lèi)型可以重寫(xiě)這個(gè)方法,我們需要將該方法修飾為virtual
意義:這個(gè)統(tǒng)一入口方法的意義在于,能夠統(tǒng)一維護(hù)觸發(fā)事件的方式,并且能夠確保事件調(diào)用的線程安全性。(避免在不同的線程觸發(fā)時(shí),事件訂閱者的狀態(tài)不同步)
示例如下:
復(fù)制代碼 代碼如下:
public class MailManager
{
public event EventHandler<NewMailEventArgs> NewMail;
protected virtual void OnNewMail(NewMailEventArgs e)
{
//處于線程安全的考慮,現(xiàn)在將對(duì)委托字段的引用復(fù)制到一個(gè)臨時(shí)字段中
EventHandler<NewMailEventArgs> temp = System.Threading.Interlocked.CompareExchange
(ref NewMail, null, null);
//如果有事件訂閱者對(duì)象的存在,則通知他們,事件已觸發(fā)
if (temp != null)
temp(this, e);
}
}
四. 在所有需要觸發(fā)事件的業(yè)務(wù)方法中,調(diào)用第三步中定義的方法
目標(biāo):在類(lèi)型中還需要有一個(gè)業(yè)務(wù)方法,來(lái)將業(yè)務(wù)中的場(chǎng)景轉(zhuǎn)化為事件觸發(fā)。。
方法:在任意需要的業(yè)務(wù)方法中,直接調(diào)用第三步的方法就可以了,不過(guò)需要實(shí)現(xiàn)封裝一個(gè)傳遞信息的類(lèi)型。
示例如下:
復(fù)制代碼 代碼如下:
public class MailManager
{
public event EventHandler<NewMailEventArgs> NewMail;
protected virtual void OnNewMail(NewMailEventArgs e)
{
//處于線程安全的考慮,現(xiàn)在將對(duì)委托字段的引用復(fù)制到一個(gè)臨時(shí)字段中
EventHandler<NewMailEventArgs> temp = System.Threading.Interlocked.CompareExchange
(ref NewMail, null, null);
//如果有事件訂閱者對(duì)象的存在,則通知他們,事件已觸發(fā)
if (temp != null)
temp(this, e);
}
public void SimulateNewMail(string from, string to, string subject)
{
//構(gòu)造一個(gè)對(duì)象來(lái)封裝向傳給事件訂閱者的信息
NewMailEventArgs e = new NewMailEventArgs(from, to, subject);
//觸發(fā)事件引發(fā)的入口方法
OnNewMail(e);
}
}
事件訂閱者類(lèi)型的設(shè)計(jì)
一. 定義類(lèi)型來(lái)訂閱和偵聽(tīng)事件
目標(biāo):設(shè)計(jì)一個(gè)傳真類(lèi)型Fax類(lèi)來(lái)偵聽(tīng)NewMail事件。
說(shuō)明:Fax類(lèi)型中需要具備對(duì)NewMail事件的訂閱和取消訂閱的方法。示例如下:
復(fù)制代碼 代碼如下:
internal sealed class Fax
{
private MailManager mailManager;
public Fax(MailManager mm)
{
this.mailManager = mm;
}
public void Register()
{
mailManager.NewMail += new EventHandler<NewMailEventArgs>(FaxMsg);
}
void FaxMsg(object sender, NewMailEventArgs e)
{
Console.WriteLine("Fax mail message");
Console.WriteLine("From = {0}, To = {1}, Subject = {2}", e.From, e.To, e.Subject);
}
public void Unregister()
{
mailManager.NewMail -= FaxMsg;
}
}
相關(guān)文章
ASP.NET FileUpload 上傳圖片實(shí)例
Add a FileUpload control to the aspx page2009-09-09
DotNetCore深入了解之HttpClientFactory類(lèi)詳解
這篇文章主要給大家介紹了關(guān)于DotNetCore深入了解之HttpClientFactory類(lèi)的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-03-03
asp.net中virtual和abstract的區(qū)別分析
這篇文章主要介紹了asp.net中virtual和abstract的區(qū)別,較為詳細(xì)的分析了virtual與abstract的概念與具體用法,并以實(shí)例的形式予以總結(jié)歸納,需要的朋友可以參考下2014-10-10
ASP.NET Page函數(shù)調(diào)用順序解析
asp.net頁(yè)面事件執(zhí)行順序,需要的朋友可以參考下。2010-06-06
asp.net調(diào)用飛信免費(fèi)發(fā)短信(測(cè)試有效)
這篇文章主要介紹了asp.net如何調(diào)用飛信免費(fèi)發(fā)短信,記得要開(kāi)通飛信把對(duì)方加為好友才能發(fā),需要的朋友可以參考下2014-05-05
.NET Core系列之MemoryCache 緩存過(guò)期
這篇文章主要介紹了.NET Core系列之MemoryCache 緩存過(guò)期,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-08-08
ASP.NET?Core?MVC中Tag?Helpers用法介紹
這篇文章介紹了ASP.NET?Core?MVC中Tag?Helpers的用法,對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-02-02
ASP.NET 緩存分析和實(shí)踐淺析提高運(yùn)行效率
說(shuō)到ASP.NET緩存,那就是:盡早緩存;經(jīng)常緩存您應(yīng)該在應(yīng)用程序的每一層都實(shí)現(xiàn)緩存。2010-02-02

