事件驅動架構Event Driven Architecture (EDA) 是一種低耦合可分布式的架構,它通常處理異步信息流。
通??梢园咽录醋魇菢I務領域中發生的一個變化。因此,在代碼中我們需要定義事件,并且實例化該事件。在.net中,事件可以看作行為的結果。必然有事件發生者和接收者,觸發事件的對象為發生者,響應事件的對象則為事件接收者。委托(delegate)則串聯起發生者和接收者。C#中委托的概念不在此累述。下面代碼是聲明委托,初始化委托和調用委托的示例。
聲明一個委托
- public delegate int TestDelegate(object obj1, object obj2) ;
實例化一個委托
- TestDelegate TD = new TestDelegate(TestDelegateMethod) ;
調用一個委托
- TestDelegateMethod(" This is a Test.");
注意,其實在C#中,更多的時候是使用event修飾的delegate。使用event修飾后的委托是一個特殊的委托,它的特殊性體現在對象的封裝性上。比如上面的代碼可以寫成
event TestDelegate TD = TestDelegateMethod;關于event和delegate的更多細節可以參考博文
http://www.tracefact.net/CSharp-Programming/Delegates-and-Events-in-CSharp.aspx
http://blog.monstuff.com/archives/000040.html
假設有如下一個應該場景,一個血壓監測儀器,在病人每一次心跳的時候,監測血壓,一旦發現病人血壓小于80,則開始報警。
那么此處,動作的發出者就是病人,一旦發出心跳這個動作后,(通知)血壓檢測儀立即檢測血壓,如果小于80,則發出警報。
有一個BloodPressureMonitor類和Paient類。BloodPressureMonitor類只用一個方法,監測血壓。病人類中主要有一個委托和一個HeartBeat()方法。其中每次調用HeartBeat()方法,都會調用委托方法。(此處,委托的意義類似于函數式編程的概念,將函數作為一個變量值,該變量賦予了哪個函數就調用哪個函數)。
因此,在main方法中,主要就是對dosth委托的賦值,也就是對病人每次心跳的時候,要調用一下血壓檢測儀的CheckPresssure方法。
代碼如下:
- public delegate void DoAction(int s);
- class BloodPressureMonitor
- {
- public void CheckPresssure(int p)
- {
- if (p < 80)
- {
- Console.WriteLine(String.Format("Alert!Bloodpressure is less than {0}", p));
- }
- else
- {
- Console.WriteLine(String.Format("Bloodpressure is {0}", p));
- }
- }
- }
- class Patient {
- public event DoAction dosth;
- public String Name { get; set; }
- public int BloodPressure { get; set; }
- public void HeartBeat()
- {
- if (dosth != null)
- dosth(BloodPressure);
- }
- }
- class Program
- {
- static void Main(string[] args)
- {
- Patient p = new Patient();
- p.Name = "Tom";
- p.BloodPressure = 100;
- BloodPressureMonitor monitor = new BloodPressureMonitor();
- p.dosth += monitor.CheckPresssure;
- for (int i = 0; i < 10; i++)
- {
- p.BloodPressure -= 5;
- p.HeartBeat();
- }
- }
- }
運行結果如下:
也許很多人會覺得,那何必這么麻煩,直接在 HeartBeat方法中調用BloodPressureMonitor的CheckPresssure方法不就行了嘛,何必用委托呢?例如直接寫成
- public void HeartBeat()
- {
- new BloodPressureMonitor(). CheckPresssure( BloodPressure );
- }
這樣的寫法不也是可以實現的嗎?
是的,但是這種寫法不靈活,首先,使用委托的話,可以利用委托的一些特點,比如調用委托的BeginInvoke方法可以異步在線程池線程上執行。
其次,如果此時再要加一個電話呼叫功能,一旦血壓小于60,電話機自動呼叫醫生,那么在現有的基礎上,改起來十分容易。只需要加個電話類,然后在委托上再掛一個方法就行了。
代碼如下:
- using System;
- using System.Collections.Generic;
- using System.Linq;
- using System.Text;
- namespace EDA
- {
- public delegate void DoAction(int s);//定義一個委托
- class BloodPressureMonitor
- {
- public void CheckPresssure(int p)
- {
- if (p < 80)
- {
- Console.WriteLine(String.Format("Alert!Bloodpressure is less than {0}", p));
- }
- else
- {
- Console.WriteLine(String.Format("Bloodpressure is {0}", p));
- }
- }
- }
- class Telephone
- {
- public void NeedCallDoctor(int p)
- {
- if (p < 60)
- {
- Console.WriteLine(String.Format("Call doctor!", p));
- }
- }
- }
- class Patient
- {
- public event DoAction dosth;
- public String Name { get; set; }
- public int BloodPressure { get; set; }
- public void HeartBeat()//心跳時,調用dosth委托
- {
- if (dosth != null)
- dosth(BloodPressure);
- }
- }
- class Program
- {
- static void Main(string[] args)
- {
- Patient p = new Patient();
- p.Name = "Tom";
- p.BloodPressure = 100;
- BloodPressureMonitor monitor = new BloodPressureMonitor();
- p.dosth += monitor.CheckPresssure;//具體設定dosth是哪些方法。用+=可以掛載多個方法
- Telephone Phone = new Telephone();
- p.dosth += Phone.NeedCallDoctor;
- for (int i = 0; i < 10; i++)
- {
- p.BloodPressure -= 5;
- p.HeartBeat();
- }
- }
- }
- }
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。