三种观察者模式的C#实现
系列主题:基于消息的软件架构模型演变
说起观察者模式,估计在园子里能搜出一堆来。所以写这篇博客的目的有两点:
- 观察者模式是写松耦合代码的必备模式,重要性不言而喻,抛开代码层面,许多组件都采用了Publish-Subscribe模式,所以我想按照自己的理解重新设计一个使用场景并把观察者模式灵活使用在其中
- 我想把C#中实现观察者模式的三个方案做一个总结,目前还没看到这样的总结
现在我们来假设这样的一个场景,并利用观察者模式实现需求:
场景:未来智能家居进入了每家每户,每个家居都留有API供客户进行自定义整合,所以第一个智能闹钟(smartClock)先登场,厂家为此闹钟提供了一组API,当设置一个闹铃时间后该闹钟会在此时做出通知,我们的智能牛奶加热器,面包烘烤机,挤牙膏设备都要订阅此闹钟闹铃消息,自动为主人准备好牛奶,面包,牙膏等。
这个场景是很典型的观察者模式,智能闹钟的闹铃是一个主题(subject),牛奶加热器,面包烘烤机,挤牙膏设备是观察者(observer),观察者只需要订阅这个主题即可实现松耦合的编码模型,让我们通过三种方案逐一实现此需求。
一、利用.net的Event模型来实现
.net中的Event模型是一种典型的观察者模型,在.net出身之后被大量应用在了代码当中,我们看事件模型如何在此种场景下使用,
首先介绍下智能闹钟,厂家提供了一组很简单的API
- public void SetAlarmTime(TimeSpan timeSpan)
- {
- _alarmTime = _now().Add(timeSpan);
- RunBackgourndRunner(_now, _alarmTime);
- }
SetAlarmTime(TimeSpan timeSpan)用来定时,当用户设置好一个时间后,闹钟会在后台跑一个类似于while(true)的循环对比时间,当闹铃时间到了后要发出一个通知事件出来
- protected void RunInBackgournd(Func<DateTime> now,DateTime? alarmTime )
- {
- if (alarmTime.HasValue)
- {
- var cancelToken = new CancellationTokenSource();
- var task = new Task(() =>
- {
- while (!cancelToken.IsCancellationRequested)
- {
- if (now.AreEquals(alarmTime.Value))
- {
- //闹铃时间到了
- ItIsTimeToAlarm();
- cancelToken.Cancel();
- }
- cancelToken.Token.WaitHandle.WaitOne(TimeSpan.FromSeconds(2));
- }
- }, cancelToken.Token, TaskCreationOptions.LongRunning);
- task.Start();
- }
- }
其他代码并不重要,重点在当闹铃时间到了后要执行ItIsTimeToAlarm(); 我们在这里发出事件以便通知观察者,.net中实现event模型有三要素,
1.为主题(subject)要定义一个event, public event Action<Clock, AlarmEventArgs> Alarm;
2.为主题(subject)的信息定义一个EventArgs,即AlarmEventArgs,这里面包含了事件所有的信息
3.主题(subject)通过以下方式发出事件
- var args = new AlarmEventArgs(_alarmTime.Value, 0.92m);
- OnAlarmEvent(args);
OnAlarmEvent方法的定义
- public virtual void OnAlarm(AlarmEventArgs e)
- {
- if(Alarm!=null)
- Alarm(this,e);
- }
这里要注意命名规范,事件内容-AlarmEventArgs,事件-Alarm(动词,例如KeyPress),触发事件的方法 void OnAlarm(),这些命名都要符合事件模型的命名规范。
智能闹钟(SmartClock)已经实现完毕,我们在牛奶加热器(MilkSchedule)中订阅这个Alarm消息:
- public void PrepareMilkInTheMorning()
- {
- _clock.Alarm += (clock, args) =>
- {
- Message ="Prepraring milk for the owner, The time is {0}, the electric quantity is {1}%".FormatWith(args.AlarmTime, args.ElectricQuantity*100);
- Console.WriteLine(Message);
- };
- _clock.SetAlarmTime(TimeSpan.FromSeconds(2));
- }
在面包烘烤机中同样可以用_clock.Alarm+=(clock,args)=>{//it is time to roast bread}订阅闹铃消息。
至此,event模型介绍完毕,实现过程还是有点繁琐的,并且事件模型使用不当会有memory leak的问题,当观察者(obsever)订阅了一个生命周期较长的主题(该主题生命周期长于观察者),该观察者将不会被垃圾回收(因为还有引用指向主题),详见Understanding and Avoiding Memory Leaks with Event Handlers and Event Aggregators,开发者需要显示退订该主题(-=)。
园子里老A也写过一篇如何利用弱引用解决该问题的博客:如何解决事件导致的Memory Leak问题:Weak Event Handlers。
二、利用.net中IObservable<out T>和IObserver<in T>实现观察者模式
IObservable<out T> 顾名思义-可观察的事物,即主题(subject),Observer很明显就是观察者了。
在我们的场景中智能闹钟是IObservable,该接口只定义了一个方法IDisposable Subscribe(IObserver<T> observer);该方法命名让人有点犯晕,Subscribe即订阅的意思,不同于之前提到过的观察者(observer)订阅主题(subject)。在这里是主题(subject)来订阅观察者(observer),其实这里也说得通,因为在该模型下,主题(subject)维护了一个观察者(observer)列表,所以有主题订阅观察者之说,我们来看闹钟的IDisposable Subscribe(IObserver<T> observer)实现:
- public IDisposable Subscribe(IObserver<AlarmData> observer)
- {
- if (!_observers.Contains(observer))
- {
- _observers.Add(observer);
- }
- return new DisposedAction(() => _observers.Remove(observer));
- }
可以看到这里维护了一个观察者列表_observers,闹钟在到点了之后会遍历所有观察者列表将消息逐一通知给观察者
- public override void ItIsTimeToAlarm()
- {
- var alarm = new AlarmData(_alarmTime.Value, 0.92m);
- _observers.ForEach(o=>o.OnNext(alarm));
- }
很明显,观察者有个OnNext方法,方法签名是一个AlarmData,代表了要通知的消息数据,接下来看看牛奶加热器的实现,牛奶加热器作为观察者(observer)当然要实现IObserver接口
- public void Subscribe(TimeSpan timeSpan)
- {
- _unSubscriber = _clock.Subscribe(this);
- _clock.SetAlarmTime(timeSpan);
- }
- public void Unsubscribe()
- {
- _unSubscriber.Dispose();
- }
- public void OnNext(AlarmData value)
- {
- Message ="Prepraring milk for the owner, The time is {0}, the electric quantity is {1}%".FormatWith(value.AlarmTime, value.ElectricQuantity * 100);
- Console.WriteLine(Message);
- }
除此之外为了方便使用面包烘烤器,我们还加了两个方法Subscribe()和Unsubscribe(),看调用过程
- var milkSchedule = new MilkSchedule();
- //Act
- milkSchedule.Subscribe(TimeSpan.FromSeconds(12));
三、Action函数式方案
在介绍该方案之前我需要说明,该方案并不是一个观察者模型,但是它却可以实现同样的功能,并且使用起来更加简练,也是我最喜欢的一种用法。
这种方案中,智能闹钟(smartClock)提供的API需要设计成这样:
- public void SetAlarmTime(TimeSpan timeSpan,Action<AlarmData> alarmAction)
- {
- _alarmTime = _now().Add(timeSpan);
- _alarmAction = alarmAction;
- RunBackgourndRunner(_now, _alarmTime);
- }
方法签名中要接受一个Action<T>,闹钟在到点后直接执行该Action<T>即可:
- public override void ItIsTimeToAlarm()
- {
- if (_alarmAction != null)
- {
- var alarmData = new AlarmData(_alarmTime.Value, 0.92m);
- _alarmAction(alarmData);
- }
- }
牛奶加热器中使用这种API也很简单:
- _clock.SetAlarmTime(TimeSpan.FromSeconds(1), (data) =>
- {
- Message ="Prepraring milk for the owner, The time is {0}, the electric quantity is {1}%".FormatWith(data.AlarmTime, data.ElectricQuantity * 100);
- });
在实际使用过程中我会把这种API设计成fluent api,调用起来代码更清晰:
智能闹钟(smartClock)中的API:
- public Clock SetAlarmTime(TimeSpan timeSpan)
- {
- _alarmTime = _now().Add(timeSpan);
- RunBackgourndRunner(_now, _alarmTime);
- return this;
- }
- public void OnAlarm(Action<AlarmData> alarmAction)
- {
- _alarmAction = alarmAction;
- }
牛奶加热器中进行调用:
- _clock.SetAlarmTime(TimeSpan.FromSeconds(2))
- .OnAlarm((data) =>
- {
- Message ="Prepraring milk for the owner, The time is {0}, the electric quantity is {1}%".FormatWith(data.AlarmTime, data.ElectricQuantity * 100);
- });
显然改进后的写法语义更好:闹钟.设置闹铃时间().当报警时(()=>{执行以下功能})
结束语:本文总结了.net下三种观察者模型的实现方案,方便大家在不同的编程场景下灵活应用最合适的模式。本文提供下载本文章所使用的源码,如需转载请注明出处
三种观察者模式的C#实现的更多相关文章
- 三种方式实现观察者模式 及 Spring中的事件编程模型
观察者模式可以说是众多设计模式中,最容易理解的设计模式之一了,观察者模式在Spring中也随处可见,面试的时候,面试官可能会问,嘿,你既然读过Spring源码,那你说说Spring中运用的设计模式吧, ...
- Java设计模式(三)——观察者模式和监听器
为了实现多个模块之间的联动,最好的方法是使用观察者模式.网上介绍的资料也比较多,今天我就从另一个方面谈谈自己对观察者模式的理解.从JDK提供的支持库里,我们能够找到四个对象:Observable.Ob ...
- PHP常用的三种设计模式
本文为大家介绍常用的三种php设计模式:单例模式.工厂模式.观察者模式,有需要的朋友可以参考下. 一.首先来看,单例模式 所谓单例模式,就是确保某个类只有一个实例,而且自行实例化并向整个系统提供这个实 ...
- ASP.NET 设计模式分为三种类型
设计模式分为三种类型,共23类. 一.创建型模式:单例模式.抽象工厂模式.建造者模式.工厂模式.原型模式. 二.结构型模式:适配器模式.桥接模式.装饰模式.组合模式.外观模式.享元模式.代 ...
- 简谈百度坐标反转至WGS84的三种思路
文章版权由作者李晓晖和博客园共有,若转载请于明显处标明出处:http://www.cnblogs.com/naaoveGIS/ 1.背景 基于百度地图进行数据展示是目前项目中常见场景,但是因为百度地图 ...
- 测试一下StringBuffer和StringBuilder及字面常量拼接三种字符串的效率
之前一篇里写过字符串常用类的三种方式<java中的字符串相关知识整理>,只不过这个只是分析并不知道他们之间会有多大的区别,或者所谓的StringBuffer能提升多少拼接效率呢?为此写个简 ...
- Objective-C三种定时器CADisplayLink / NSTimer / GCD的使用
OC中的三种定时器:CADisplayLink.NSTimer.GCD 我们先来看看CADiskplayLink, 点进头文件里面看看, 用注释来说明下 @interface CADisplayLin ...
- css中的浮动与三种清除浮动的方法
说到浮动之前,先说一下CSS中margin属性的两种特殊现象 1, 外边距的合并现象: 如果两个div上下排序,给上面一个div设置margin-bottom,给下面一个div设置margin-top ...
- ASP.NET Core应用的错误处理[1]:三种呈现错误页面的方式
由于ASP.NET Core应用是一个同时处理多个请求的服务器应用,所以在处理某个请求过程中抛出的异常并不会导致整个应用的终止.出于安全方面的考量,为了避免敏感信息的外泄,客户端在默认的情况下并不会得 ...
随机推荐
- 【leetcode】Path Sum
题目简述: Given a binary tree and a sum, determine if the tree has a root-to-leaf path such that adding ...
- 前端试题本(Javascript篇)
JS1. 下面这个JS程序的输出是什么:JS2.下面的JS程序输出是什么:JS3.页面有一个按钮button id为 button1,通过原生的js如何禁用?JS4.页面有一个按钮button id为 ...
- Java 之 I/O流
1.流 a.分类:①字节流:InputStream.OutputStream ②字符流:Reader.Writer b.选择:①判断是 输入 还是 输出 (站在程序的立场上) ②判断是 字节 还是 字 ...
- SNMP高速扫描器braa
SNMP高速扫描器braa SNMP(Simple Network Monitoring Protocol,简单网络管理协议)是网络设备管理标准协议.为了便于设备管理,现在联入网络的智能设备都支持 ...
- mysql5.7绿色版安装与配置
1,找到zip archive包下载,官方地址如下: http://dev.mysql.com/downloads/mysql/ http://dev.mysql.com/downloads/file ...
- select2 清空数据
最近用select2插件,发现用jquery重置不好使,最后搜罗了一把发现下面这个方法可以间接的实现,有空还得看看插件的API $('#integratorId').select2('data', n ...
- 建立jackrabbit内容仓库实例
jackrabbit需要内容仓库主目录和内容仓库配置文件这两部分的信息才能创建一个运行时内容仓库实例. 1.内容仓库主目录结构 2.Repository.xml的配置文件结构
- 【转】OBJECT_ID和DATA_OBJECT_ID的区别
在user_objects等视图里面有两个比较容易搞混的字段object_id和data_object_id这两个字段基本上有什么大的区别呢?object_id其实是对每个数据库中数据对象的唯一标识d ...
- 体育游戏中的Player类
最近在做一个棒球的游戏,开始感觉还是挺酷炫的,但是其实做法挺朴实的,想象中的球员是多么智能,这样那样的,其实只是表象. 关于球员的类是游戏里非常重要的部分,这个玩意怎么写呢,可以这样写...... 棒 ...
- How to make your assembly more secure from referencing by unauthorized bits
Now the security has a trend to become more and more important in our daily work, hence I did some r ...