成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

C#事件的由來

開發 后端
本文講述了C#事件的由來,希望對大家有所幫助。

在介紹C#事件的由來之前,我們繼續思考上面的程序:上面的三個方法都定義在Programe類中,這樣做是為了理解的方便,實際應用中,通常都是 GreetPeople 在一個類中,ChineseGreeting和 EnglishGreeting 在另外的類中。現在你已經對委托有了初步了解,是時候對上面的例子做個改進了。假設我們將GreetingPeople()放在一個叫GreetingManager的類中,那么新程序應該是這個樣子的:

  1. namespace Delegate {  
  2.     //定義委托,它定義了可以代表的方法的類型  
  3.     public delegate void GreetingDelegate(string name);  
  4.       
  5.     //新建的GreetingManager類  
  6.     public class GreetingManager{  
  7.        public void GreetPeople(string name, GreetingDelegate MakeGreeting) {  
  8.            MakeGreeting(name);  
  9.        }  
  10.     }  
  11.  
  12.     class Program {  
  13.        private static void EnglishGreeting(string name) {  
  14.            Console.WriteLine("Morning, " + name);  
  15.        }  
  16.  
  17.        private static void ChineseGreeting(string name) {  
  18.            Console.WriteLine("早上好, " + name);  
  19.        }  
  20.  
  21.        static void Main(string[] args) {  
  22.            // ... ...  
  23.         }  
  24.     }  

這個時候,如果要實現前面演示的輸出效果,Main方法我想應該是這樣的:

  1. static void Main(string[] args) {  
  2.     GreetingManager gm = new  GreetingManager();  
  3.     gm.GreetPeople("Jimmy Zhang", EnglishGreeting);  
  4.     gm.GreetPeople("張子陽", ChineseGreeting);  

我們運行這段代碼,嗯,沒有任何問題。程序一如預料地那樣輸出了:

  1. Morning, Jimmy Zhang  
  2.  
  3. 早上好, 張子陽 

現在,假設我們需要使用上一節學到的知識,將多個方法綁定到同一個委托變量,該如何做呢?讓我們再次改寫代碼:

  1. static void Main(string[] args) {  
  2.     GreetingManager gm = new  GreetingManager();  
  3.     GreetingDelegate delegate1;  
  4.     delegate1 = EnglishGreeting;  
  5.     delegate1 += ChineseGreeting;  
  6.  
  7.     gm.GreetPeople("Jimmy Zhang", delegate1);  

輸出:

  1. Morning, Jimmy Zhang  
  2. 早上好, Jimmy Zhang 

到了這里,我們不禁想到:面向對象設計,講究的是對象的封裝,既然可以聲明委托類型的變量(在上例中是delegate1),我們何不將這個變量封裝到 GreetManager類中?在這個類的客戶端中使用不是更方便么?于是,我們改寫GreetManager類,像這樣:

  1. public class GreetingManager{  
  2.     //在GreetingManager類的內部聲明delegate1變量  
  3.     public GreetingDelegate delegate1;    
  4.  
  5.     public void GreetPeople(string name, GreetingDelegate MakeGreeting) {  
  6.        MakeGreeting(name);  
  7.     }  

現在,我們可以這樣使用這個委托變量:

  1. static void Main(string[] args) {  
  2.     GreetingManager gm = new  GreetingManager();  
  3.     gm.delegate1 = EnglishGreeting;  
  4.     gm.delegate1 += ChineseGreeting;  
  5.  
  6.     gm.GreetPeople("Jimmy Zhang", gm.delegate1);  

輸出為:

  1. Morning, Jimmy Zhang  
  2. 早上好, Jimmy Zhang 

盡管這樣做沒有任何問題,但我們發現這條語句很奇怪。在調用gm.GreetPeople方法的時候,再次傳遞了gm的delegate1字段:

  1. gm.GreetPeople("Jimmy Zhang", gm.delegate1); 

既然如此,我們何不修改 GreetingManager 類成這樣:

  1. public class GreetingManager{  
  2.     //在GreetingManager類的內部聲明delegate1變量  
  3.     public GreetingDelegate delegate1;    
  4.  
  5.     public void GreetPeople(string name) {  
  6.         if(delegate1!=null){     //如果有方法注冊委托變量  
  7.           delegate1(name);      //通過委托調用方法  
  8.        }  
  9.     }  

在客戶端,調用看上去更簡潔一些:

  1. static void Main(string[] args) {  
  2.     GreetingManager gm = new  GreetingManager();  
  3.     gm.delegate1 = EnglishGreeting;  
  4.     gm.delegate1 += ChineseGreeting;  
  5.  
  6.     gm.GreetPeople("Jimmy Zhang");      //注意,這次不需要再傳遞 delegate1變量  

輸出為:

  1. Morning, Jimmy Zhang  
  2. 早上好, Jimmy Zhang 

盡管這樣達到了我們要的效果,但是還是存在著問題:

在這里,delegate1和我們平時用的string類型的變量沒有什么分別,而我們知道,并不是所有的字段都應該聲明成public,合適的做法是應該public的時候public,應該private的時候private。

我們先看看如果把 delegate1 聲明為 private會怎樣?結果就是:這簡直就是在搞笑。因為聲明委托的目的就是為了把它暴露在類的客戶端進行方法的注冊,你把它聲明為private了,客戶端對它根本就不可見,那它還有什么用?

再看看把delegate1 聲明為 public 會怎樣?結果就是:在客戶端可以對它進行隨意的賦值等操作,嚴重破壞對象的封裝性。

***,***個方法注冊用“=”,是賦值語法,因為要進行實例化,第二個方法注冊則用的是“+=”。但是,不管是賦值還是注冊,都是將方法綁定到委托上,除了調用時先后順序不同,再沒有任何的分別,這樣不是讓人覺得很別扭么?

現在我們想想,如果delegate1不是一個委托類型,而是一個string類型,你會怎么做?答案是使用屬性對字段進行封裝。

于是,Event出場了,它封裝了委托類型的變量,使得:在類的內部,不管你聲明它是public還是protected,它總是private的。在類的外部,注冊“+=”和注銷“-=”的訪問限定符與你在聲明事件時使用的訪問符相同。

我們改寫GreetingManager類,它變成了這個樣子:

  1. public class GreetingManager{  
  2.     //這一次我們在這里聲明一個事件  
  3.     public event GreetingDelegate MakeGreet;  
  4.  
  5.     public void GreetPeople(string name) {  
  6.         MakeGreet(name);  
  7.     }  

很容易注意到:MakeGreet 事件的聲明與之前委托變量delegate1的聲明唯一的區別是多了一個event關鍵字。看到這里,在結合上面的講解,你應該明白到:C#事件事件其實沒什么不好理解的,聲明一個事件不過類似于聲明一個進行了封裝的委托類型的變量而已。

為了證明上面的推論,如果我們像下面這樣改寫Main方法:

  1. static void Main(string[] args) {  
  2.     GreetingManager gm = new  GreetingManager();  
  3.     gm.MakeGreet = EnglishGreeting;         // 編譯錯誤1  
  4.     gm.MakeGreet += ChineseGreeting;  
  5.  
  6.     gm.GreetPeople("Jimmy Zhang");  

會得到編譯錯誤:事件“Delegate.GreetingManager.MakeGreet”只能出現在 += 或 -= 的左邊(從類型“Delegate.GreetingManager”中使用時除外)。

以上就是C#事件的由來。

【編輯推薦】

  1. C#調用SQL存儲過程示例演示
  2. C#與SQL連接:GridView控件操作
  3. ASP調用C# DLL發送郵件方法共享
  4. VC6.0和VS2005:C++和C#編寫調用COM組件
  5. C#中調用dll庫(VC編寫)
責任編輯:book05 來源: cnblogs
相關推薦

2009-08-07 17:57:26

C#定義事件應用

2009-08-28 09:43:05

C#事件

2009-08-12 15:20:21

C#事件處理

2009-08-31 18:01:41

C#接口事件

2024-05-16 13:36:04

C#委托事件

2009-08-28 13:12:02

C# Page_Err

2009-08-27 18:02:22

C#事件處理

2009-08-13 10:42:31

C#窗體拖動事件

2009-09-07 15:31:49

C#支持事件

2011-04-25 09:22:44

C#事件

2009-09-07 04:19:56

C#窗體事件

2009-08-28 14:28:25

C# Applicat

2009-09-10 18:18:42

C# Button

2009-08-12 18:28:09

C#事件處理程序

2009-08-03 13:23:04

C#編程組件-事件-委托

2024-05-15 09:11:51

委托事件C#

2009-08-18 10:54:17

C#事件和委托

2024-06-28 10:19:02

委托事件C#

2009-08-04 09:56:46

C#事件處理自定義事件

2009-08-04 13:53:58

C#委托類C#事件
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 日韩视频在线免费观看 | 久久国产一区二区 | 成人片免费看 | 国产jizz女人多喷水99 | 一区视频在线免费观看 | 草比网站| 国产三区在线观看视频 | 国产精品久久久久久久久免费高清 | 国产精品久久久久一区二区三区 | 国产一区二区影院 | 精品久久久久久亚洲综合网 | 日韩在线观看网站 | 国产精品18久久久久久白浆动漫 | 国产精品美女在线观看 | va在线 | 日韩国产高清在线观看 | 成人九区 | 欧美一级免费 | 国产一区不卡在线观看 | 欧美精品在线免费观看 | 欧美中文字幕 | 青青久久| 日本福利在线 | 成人在线免费电影 | 久久国产精品免费一区二区三区 | 日韩视频一区二区三区 | 亚洲综合天堂网 | 欧美一区二区三区 | 做a视频| 色偷偷噜噜噜亚洲男人 | 日日夜夜精品 | 亚洲精品乱码久久久久久9色 | 一区二区不卡视频 | 蜜臀91视频 | 亚洲天堂免费在线 | 玖玖精品 | 中文字幕一级 | 一本一道久久a久久精品蜜桃 | 亚洲第1页 | 91在线精品一区二区 | 国产日韩欧美另类 |