.Net邊學邊講(三)

來源:互聯網
上載者:User
談到event,就不能不先說一下callback和delegate
如果你使用過C的話,你應該知道有一個函數叫qsort,是用來給數組排序的。但這個函數顯然不能承擔廣泛意義上的比較,因此你需要傳遞一個指標,他指向具有比較功能的函數。qsort在每次要比較數組元素時都要調用這個函數。這就是callback的概念,在.Net裡也可以實現回調,方法是建立一個介面,實現他,傳遞一個實現此介面的對象的引用。delegate呢,你可以將他理解成一個安全的函數指標。
Notifications跟callback有點類似,但比簡單的回調要複雜的多。callback意味著要調用的callback方法被調用的同時要調用的建立callback的方法。這是一個很緊密的耦合。而Notification則要鬆散一些,你可以註冊將來某段時間會或者不會發生的Notification,只當他們發生時處理,否則不用。
你也許想讓你寫的組件當一些事情發生時通知其他組件,例如,你想寫一個按鈕組件,當你Click的時候你可能想通知其他組件,而其他組件將不得不準備向你請求Notification,你就要提供一個方法告訴他們你已經有一個可用的Notification。另一方面,你可能也是當其它組件的一些事情發生時希望被通知的人。這時你就需要找到那個特定組件可以提供什麼Notification。
在.Net中event是一個你用來廣播、引發、處理Notification的機制。大致是這樣的,可以引發事件的組件聲明這一事件。而希望處理某一特定組件的某一特定事件的組件通過傳遞一個方法的delegate向引發事件的組件註冊。這樣,當事件發生時引發事件的組件就會調用每個登入的方法。通過delegate和event我們可以實現非同步呼叫的功能。在C#中是這樣聲明一個代理的:public delegate void LogHandler(String message);代理在處理這種回調時已經是很強大了。但是當我們需要代理被儲存以便以後的Notification,就有一點麻煩了。比如說我們有一個對象Button,有一個Click事件。我們可以聲明一個ClickHandler 的代理類型用於處理Click事件,在我們的Button的Class中聲明一個ClickHandler的public執行個體,這樣其他組件希望Click發生時被通知,就可以簡單的把他的代理加到Click代理中去。myButton.Click += new ClickHandler(MyMethod);
看上去著好像沒什麼問題。但是這裡卻存在一個大問題,我們聲明Click代理是public,這違反了我們以前說過的data fields永遠不要聲明成public,這會有一系列麻煩。解決的辦法是聲明成private或protected,然後用屬性解決讀寫。這樣我們可以private聲明Click,在寫一對public方法去增加一個listener及減少一個listener。當然在.Net中,當你聲明一個event時,.Net已經為你做好這一切了。聲明一個事件:
class AlarmTimer {
   public event EventHandler Alarm;
   // ...
}
這段代碼說明AlarmTimer可以向所有其它對象廣播它可以引發一個叫Alarm的事件。Alarm事件用的是EventHandler代理類型。EventHandler:無傳回值、接受兩個參數(Object:指向事件的寄件者,EventArgs:包含關於事件的資料)
看一個例子:
class AlarmTimer {
   public event EventHandler Alarm;
   private Timer myTimer;
   public AlarmTimer() {
      myTimer = new Timer();
      myTimer.Tick += new EventHandler(OnTick);
   }
   public void Set(Double seconds) {
      myTimer.Interval = (Int32)(seconds * 1000);
      myTimer.Start();
   }
   protected void OnTick(Object sender, EventArgs e) {
      myTimer.Stop();
      if (Alarm != null) Alarm(this, EventArgs.Empty);
   }
   public void ReEnable() {
      myTimer.Enabled = true;
   }
}
注意AlarmTimer既引發事件(Alarm)又處理事件(Timer中的Tick事件)
static AlarmTimer myAlarm = new AlarmTimer();
public static void TestEvent() {
   myAlarm.Alarm += new EventHandler(TimerEventProcessor);
   myAlarm.Set(2);
   Console.WriteLine("Timer is set; alarm will go off in two seconds");        
   Application.Run();
}
//處理事件
private static void TimerEventProcessor(
Object myObject, EventArgs myEventArgs) {
   if (MessageBox.Show("Wake up! Continue ringing?",
"Count is: " + alarmCounter,
MessageBox.YesNo) == DialogResult.Yes) {
      alarmCounter += 1;
      myAlarm.ReEnable();
   }
   else {
      Application.Exit();
   }
}
注意:Object和EventArgs不是必要參數,只是這是一個好的寫法模式而已
關於event還有好多沒有說,留著以後慢慢說吧。

相關文章

E-Commerce Solutions

Leverage the same tools powering the Alibaba Ecosystem

Learn more >

Apsara Conference 2019

The Rise of Data Intelligence, September 25th - 27th, Hangzhou, China

Learn more >

Alibaba Cloud Free Trial

Learn and experience the power of Alibaba Cloud with a free trial worth $300-1200 USD

Learn more >

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。