事件驱动架构(EDA)和观察者模式
事件驱动架构Event Driven Architecture (EDA) 是一种低耦合可分布式的架构,它通常处理异步信息流。
公司主营业务:成都网站设计、成都网站制作、外贸网站建设、移动网站开发等业务。帮助企业客户真正实现互联网宣传,提高企业的竞争能力。创新互联建站是一支青春激扬、勤奋敬业、活力青春激扬、勤奋敬业、活力澎湃、和谐高效的团队。公司秉承以“开放、自由、严谨、自律”为核心的企业文化,感谢他们对我们的高要求,感谢他们从不同领域给我们带来的挑战,让我们激情的团队有机会用头脑与智慧不断的给客户带来惊喜。创新互联建站推出明溪免费做网站回馈大家。
通常可以把事件看作是业务领域中发生的一个变化。因此,在代码中我们需要定义事件,并且实例化该事件。在.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();
- }
- }
- }
- }
当前文章:事件驱动架构(EDA)和观察者模式
本文链接:http://hbruida.cn/article/iphjig.html