首先來說,委托的作用就是可以給類的方法傳遞其他類的方法而不必將類實例化。第二點,委托就是事件和響應事件的方法的橋梁(就是傳遞響應事件的方法給事件)。這裡要註意,委托跟類平級,事件與方法平級。 全文 事件與委托似乎很難以理解,這是因為它們的使用方式與常用的編碼有很大的差別,例如通常編寫的都是同步代碼, ...
首先來說,委托的作用就是可以給類的方法傳遞其他類的方法而不必將類實例化。第二點,委托就是事件和響應事件的方法的橋梁(就是傳遞響應事件的方法給事件)。這裡要註意,委托跟類平級,事件與方法平級。
全文
事件與委托似乎很難以理解,這是因為它們的使用方式與常用的編碼有很大的差別,例如通常編寫的都是同步代碼,調用一個類型的方法,會即刻出現方法執行的結果,這是符合邏輯的。但在某些情況中,同步代碼未必滿足需求,拿公車來打個比方,如果交通管制中心希望每一輛公車到達一個站點時都發送給自己一個信號以便自己能夠隨時掌握交通狀況,使用同步代碼,公汽對象肯定需要調用管制中心對象,這樣就出現了我們一直不願意看到的情況:兩個類型緊密地耦合在一起。既然要其它類型對自己的行為作出反應,親自調用其類型的方法似乎不可避免,在同步代碼中,很難避免這種緊密的類型調用關係。 另一個差別是在一般情況下,我們只將屬性作為參數傳遞給方法,而很少會考慮將一個方法傳遞給另一個方法。 我們拋棄各種C#參考書中桀驁難懂的事件與委托概念,設想一個情景來理解事件與委托的使用:有一家IT公司,董事長不希望自己的雇員在上班時間玩游戲,但又不可能每時每刻都盯著每個雇員,因此,他希望使用一種新的方式實現監視雇員的效果:如果有雇員違反規定,某個設備或專門的監查人員將自動發出一個消息通知他,董事長只需要在事情發生時進行處理。 因此,這個用例實際上是兩種類型——董事長類與雇員類——之間的交互,下麵的代碼將給讀者展示如何使用委托與事件機制實現這種交互: 首先,我們需要在董事長類與雇員類之間定義一個委托類型,用於傳遞兩者之間的事件,這個類型就是一個監視設備或專門負責打小報告的監查人員:
- public delegate void DelegateClassHandle();
定義一個委托的過程類似方法的定義,但它沒有方法體。定義委托一定要添加關鍵字delegate。由於定義委托實際上相當一個類,因此可以在定義類的任何地方定義委托。另外,根據委托的可見性,也可以添加一般的訪問修飾符,如public、private和protected。 委托的返回值類型為void,這並非表示委托類型本身帶有返回值,該返回值類型是指委托的目標函數類型,即它委托的一個事件處理函數返回值是void類型。 新建一個雇員類Employee,其代碼如下:
- public class Employee
- {
- public event DelegateClassHandle PlayGame;
- public void Games()
- {
- if (PlayGame != null)
- {
- PlayGame();
- }
- }
- }
雇員類Employee代碼中定義了一個DelegateClassHandle類型的事件PlayGame,它的定義方式也很特殊,首先必須使用關鍵字event,表示PlayGame是一個事件,同時還必須聲明該事件的委托類型為DelegateClassHandle,即將來由該類型的委托對象負責通知事件。 如果有雇員開始玩游戲,它將執行Games方法,而只要該方法一被調用,就會觸發一個事件PlayGame,然後董事長就會收到這個事件的消息——有人在玩游戲了。 董事長類代碼如下,他有一個方法Notify用於接收消息:
- public class Admin
- {
- public void Notify()
- {
- System.Console.WriteLine("someone is playing game");
- }
- }
Employee的PlayGame事件如何與Admin的Notify方法關聯起來呢?只需通過事件綁定即可實現,具體過程如下列代碼:
- Employee employee = new Employee();
- Admin admin = new Admin();
- employee.PlayGame += new DelegateClassHandle(admin.Notify);
- employee.Games();
請大家註意事件綁定的代碼:
- employee.PlayGame += new DelegateClassHandle(admin.Notify);
通過DelegateClassHandle將兩個類的交互進行了綁定,當employee.Games方法調用後,觸發PlayGame事件,而該事件將被委托給admin的Notify方法處理,通知董事長有雇員在上班時間玩游戲。 但董事長並不滿足這種簡單的通知,他還想知道究竟是誰在上班時間違反規定。顯然,現在委托對象必須傳遞必要的參數才行,這個要求也可以很容易地辦到。事件的參數可以設置為任何類型的數據,在.NET框架中,還提供了事件參數基類EventArgs專門用於傳遞事件數據。 從該EventArgs類派生一個自定義的事件參數類CustomeEventArgs,這個類型將攜帶雇員姓名和年齡信息:
- public class CustomeEvetnArgs : EventArgs
- {
- string name = "";
- int age = 0;
- public CustomeEvetnArgs()
- { }
- publicstring Name
- {
- get { returnthis.name; }
- set { this.name = value; }
- }
- publicint Age
- {
- get { returnthis.age; }
- set { this.age = value; }
- }
- }
修改委托類型DelegateClassHandle的定義,讓其攜帶必要的參數: public delegate void DelegateClassHandle(object sender, CustomeEvetnArgs e); 雇員類的代碼修改後如下:
- public class Employee
- {
- private string _name;
- public string Name
- {
- get { return _name; }
- set { _name = value; }
- }
- private int _age;
- public int Age
- {
- get { return _age; }
- set { _age = value; }
- }
- public event DelegateClassHandle PlayGame;
- public void Games()
- {
- if (PlayGame != null)
- {
- CustomeEvetnArgs e = new CustomeEvetnArgs();
- e.Name = this._name ;
- e.Age = this._age;
- PlayGame(this, e);
- }
- }
- }
在Games方法中,首先新建一個CustomeEventArgs對象,然後設置了必要的屬性Name和Age。 董事長的通知方法也必須相應地進行修改:
- public class Admin
- {
- public void Notify(object sender, CustomeEvetnArgs e)
- {
- System.Console.WriteLine(e.Name+" is "+e.Age.ToString());
- }
- }
將兩個類型對象進行關聯的代碼也需要進行相應的修改:
- Employee employee = new Employee();
- employee.Name = "Mike";
- employee.Age = 25;
- Admin admin = new Admin();
- employee.PlayGame += new DelegateClassHandle(admin.Notify);
- employee.Games();
修改後的代碼運行的結果是,當Mike調用Games方法玩游戲時,會自動觸發PlayGame事件,而該事件攜帶相關信息通知admin,後者的Notify方法將接收到數據並輸出“Mike is 25”,告訴董事長是Mike,25歲,正在上班時間玩游戲。 委托是可以多路廣播(Mulitcast)的,即一個事件可以委托給多個對象接收並處理。在上面的用例中,如果有另一位經理與董事長具有同樣的癖好,也可以讓委托對象將雇員的PlayGame事件通知他。 首先定義經理類:
- public class Manager
- {
- public void Notify(object sender, CustomeEvetnArgs e)
- {
- System.Console.WriteLine(sender.ToString() + "-" + e.Name);
- }
- }
經理Manager類型的Notify方法與Admin一致,他也接受到相應的信息。 委托的多路廣播綁定的方法仍然是使用+=運算符,其方法如下麵的代碼所示:
- Employee employee = new Employee();
- employee.Name = "Mike";
- employee.Age = 25;
- Admin admin = new Admin();
- Manager manager = new Manager();
- employee.PlayGame += new DelegateClassHandle(admin.Notify);
- employee.PlayGame += new DelegateClassHandle(manager.Notify);
- employee.Games();
執行該方法,讀者將看到admin和manager的Notify方法都會被事件通知並調用執行。通過這樣的方法,董事長和經理都會知道Mike在玩游戲了。 如果董事長不希望經理也收到這個通知,該如何解除PlayGame對manager的事件綁定呢?同樣非常簡單,在employee.Games方法被調用前執行下列語句即可:
- employee.PlayGame -= new DelegateClassHandle(manager.Notify);
最後需要提醒讀者註意的,Employee類中的Games方法在觸發事件PlayGame之前需要判斷該事件是否為null。當employee對象的 Games方法觸發事件PlayGame後,必須有一個目標函數來處理這個事件,而該語句正是判斷該目標函數是否存在。如果將這個判斷去掉,且對事件不進行任何綁定而直接調用Games方法,程式將在事件PlayGame處彈出一個NullReferenceException的異常。 讀者能夠從委托與事件的代碼中得出什麼結論嗎?兩個需要存在調用關係的類型,在各自的實現中卻沒有編寫實際的調用代碼,它們只是通過一個事件和一個第三方的委托類型完成了消息的傳遞過程。兩個類型之間不存在任何的緊密耦合,它們看似鬆散地通過一個委托對象中通信,實現了本書一直宣傳的“高聚合”和“低耦合”觀點。
分類: C#課堂