设计模式学习之“观察者模式” [C#]
《深入浅出设计模式》学习笔记第二章
需求:
开发一套气象监测应用,如图:

气象站,目前有三种装置,温度、湿度和气压感应装置。
WeatherData对象追踪气象站的数据,并更新到布告板,布告板(目前是三个:目前状况、气象统计、天气预报)用来显示目前的天气状况给用户。
初步设计

目前的要求:
1.其中有三个方法分别获得气温、湿度和气压的数据。
2.一旦气象测量被更新,那么这个measurementsChanged()方法就会被调用。

3.一旦有新的数据,者三个布告板(暂时三个)就会马上更新。
4.可扩展,可以开发第三方的布告板。
错误的实现:

错误在哪:
1.变化的地方需要封装。
2.布告板应该统一实现某个带有update方法的接口。
3.不应该针对实现编程,应该针对接口编程。
什么是观察者模式 Observer Pattern
例子:
我们订阅公众号,公众号一旦有新文章就会发送给我们。
当我不再想看文章时,就取消订阅,这时就不会给我发送文章了。
只要公众号还在运营,就一直有人订阅或者取消订阅。
出版者(Publishers) + 订阅者(Subscribers) = 观察者模式(Observer Pattern)
不过我们用的名词不一样,出版者改为主题(Subject),订阅者改为观察者(Observer)
观察者模式定义:
观察者模式定义了对象之间的一对多依赖,这样一来,当一个对象改变状态时,它的所有依赖者(dependents)都会收到通知并自动更新。

观察这模式关系图:

松耦合
两个对象之间的松耦合就是,他们可以交互,但是不清楚对方太多细节。
观察者模式的Subject与Observers之间是松耦合,因为:
1.Subject对于Observer知道的唯一一件事情就是它实现了某个接口。
2.随时可以添加新的Observer。
3.新的Observer出现时,永远不需要更改Subject的代码。
4.我们可以独立的复用Subject或Observer。
5.改变任意的一方并不会影响另外一方。
设计原则
交互对象之间应该尽可能的去实现松耦合设计。
气象应用的具体设计:

C#实现:
接口们:
namespace C02ObserverPattern.Raw.Bases
{
public interface ISubject
{
// 订阅
void RegisterObserver(IObserver observer);
// 取消订阅
void RemoveObserver(IObserver observer);
// 状态变化时,通知所有观察者
void NotifyObservers();
}
public interface IObserver
{
// 气象之变化时,subject会把这些值更新给observers
void Update(float temp, float humidity, float pressure);
}
public interface IDisplayElement
{
// 布告板显示
void Display();
}
}
WeatherData(subject):
namespace C02ObserverPattern.Raw
{
public class WeatherData: ISubject
{
private readonly HashSet<IObserver> _observers;
private float _temp, _humidity, _pressure;
public WeatherData()
{
_observers = new HashSet<IObserver>();
}
public void RegisterObserver(IObserver observer)
{
_observers.Add(observer);
}
public void RemoveObserver(IObserver observer)
{
_observers.Remove(observer);
}
public void NotifyObservers()
{
foreach (var observer in _observers)
{
observer.Update(_temp, _humidity, _pressure);
}
}
public void MeasurementsChanged()
{
NotifyObservers();
}
public void SetMeasurements(float temp, float humidity, float pressure)
{
_temp = temp;
_humidity = humidity;
_pressure = pressure;
MeasurementsChanged();
}
}
}
其中一个布告板:
namespace C02ObserverPattern.Raw
{
public class CurrentConditionDisplay: IObserver, IDisplayElement
{
private float _temp, _humidity;
public CurrentConditionDisplay(ISubject weatherData)
{
weatherData.RegisterObserver(this);
}
public void Update(float temp, float humidity, float pressure)
{
_temp = temp;
_humidity = humidity;
Display();
}
// 这个布告板只显示温度和湿度
public void Display()
{
Console.WriteLine($"Current conditions:{_temp}℃ and {_humidity}% humidity");
}
}
}
测试程序:
namespace C02ObserverPattern
{
class Program
{
static void Main(string[] args)
{
WeatherData weatherData = new WeatherData();
CurrentConditionDisplay currentConditionDisplay = new CurrentConditionDisplay(weatherData);
StatisticsDisplay statisticsDisplay = new StatisticsDisplay(weatherData);
// XxxDisplay1 xxxDisplay1 = new XxxDisplay1(weatherData);
// XxxDisplay2 xxxDisplay2 = new XxxDisplay2(weatherData);
weatherData.SetMeasurements(, , );
weatherData.SetMeasurements(, , );
weatherData.SetMeasurements(, , );
Console.ReadLine();
}
}
}
结果:

----------------------------------------------------------------------------------------------------------------------------------------------------------------
使用C#内置的Observer Pattern
IObservable<T> 作为Subject。
IObserver<T> 作为Observer。
先封装一下数据:
namespace C02ObserverPattern.BuiltIn
{
public struct MyData
{
public float Temperature { get; }
public float Humidity { get; }
public float Pressure { get; }
public MyData(float temperature, float humidity, float pressure)
{
Temperature = temperature;
Humidity = humidity;
Pressure = pressure;
}
}
}
WeatherData:
namespace C02ObserverPattern.BuiltIn
{
public class WeatherDataAlternative : IObservable<MyData>
{
private readonly HashSet<IObserver<MyData>> _observers;
public WeatherDataAlternative()
{
_observers = new HashSet<IObserver<MyData>>();
}
public IDisposable Subscribe(IObserver<MyData> observer)
{
if (!_observers.Contains(observer))
{
_observers.Add(observer);
}
return new Unsubscriber(_observers, observer);
}
public void NotifyMeasurementsChanged(MyData? data)
{
foreach (var observer in _observers)
{
if (!data.HasValue)
{
observer.OnError(new Exception("No value!!!"));
}
else
{
observer.OnNext(data.Value);
}
}
}
public void WeatherStationClose()
{
foreach (var observer in _observers.ToArray())
{
observer.OnCompleted();
}
_observers.Clear();
Console.ForegroundColor = ConsoleColor.Red;
Console.WriteLine("The weather station has been closed.");
}
private class Unsubscriber : IDisposable
{
private readonly HashSet<IObserver<MyData>> _observers;
private readonly IObserver<MyData> _observer;
public Unsubscriber(HashSet<IObserver<MyData>> observers, IObserver<MyData> observer)
{
_observers = observers;
_observer = observer;
}
public void Dispose()
{
if (_observer != null && _observers.Contains(_observer))
_observers.Remove(_observer);
}
}
}
}
实现两个布告板:
namespace C02ObserverPattern.BuiltIn
{
public class CurrentConditionDisplayAlternative : IObserver<MyData>
{
private IDisposable _unsubscriber;
public virtual void Subscribe(IObservable<MyData> provider)
{
if (provider != null)
_unsubscriber = provider.Subscribe(this);
}
public void OnNext(MyData value)
{
Console.WriteLine($"Current condition: {value.Temperature}C degrees and {value.Humidity}% humidity and {value.Pressure} pressure.");
}
public void OnError(Exception error)
{
var original = Console.ForegroundColor;
Console.ForegroundColor = ConsoleColor.Red;
Console.WriteLine("Error occurred at Current Condition Display!");
Console.ForegroundColor = original;
}
public void OnCompleted()
{
Console.WriteLine("Current condition display Task Completed.");
Unsubscribe();
}
public virtual void Unsubscribe()
{
Console.WriteLine("Current condition display will unsubscribe from weather station.");
_unsubscriber.Dispose();
}
}
}
namespace C02ObserverPattern.BuiltIn
{
public class StatisticsDisplayAlternative : IObserver<MyData>
{
private IDisposable _unsubscriber;
public virtual void Subscribe(IObservable<MyData> provider)
{
if (provider != null)
_unsubscriber = provider.Subscribe(this);
}
public void OnNext(MyData value)
{
Console.WriteLine($"Statistics: {value.Temperature}, {value.Humidity}, {value.Pressure}.");
}
public void OnError(Exception error)
{
var original = Console.ForegroundColor;
Console.ForegroundColor = ConsoleColor.Red;
Console.WriteLine("Error occurred at Statistics!");
Console.ForegroundColor = original;
}
public void OnCompleted()
{
Console.WriteLine("Statistics Task Completed.");
Unsubscribe();
}
public virtual void Unsubscribe()
{
Console.WriteLine("Statistics will unsubscribe from weather station.");
_unsubscriber.Dispose();
}
}
}
测试程序:
namespace C02ObserverPattern
{
class Program
{
static void Main(string[] args)
{
WeatherDataAlternative weatherDataAlternative = new WeatherDataAlternative();
CurrentConditionDisplayAlternative currentConditionDisplayAlternative = new CurrentConditionDisplayAlternative();
currentConditionDisplayAlternative.Subscribe(weatherDataAlternative);
StatisticsDisplayAlternative statisticsDisplayAlternative = new StatisticsDisplayAlternative();
statisticsDisplayAlternative.Subscribe(weatherDataAlternative);
weatherDataAlternative.NotifyMeasurementsChanged(, , ));
Task.Delay();
weatherDataAlternative.NotifyMeasurementsChanged(null);
Task.Delay();
currentConditionDisplayAlternative.Unsubscribe();
weatherDataAlternative.NotifyMeasurementsChanged(, , ));
weatherDataAlternative.WeatherStationClose();
Console.ReadLine();
}
}
}
设计模式学习之“观察者模式” [C#]的更多相关文章
- Java设计模式学习记录-观察者模式
前言 观察者模式也是对象行为模式的一种,又叫做发表-订阅(Publish/Subscribe)模式.模型-视图(Model/View)模式. 咱们目前用的最多的就是各种MQ(Message Queue ...
- 【C++深入浅出】设计模式学习之观察者模式
前言 前两天学习了weak_ptr以后还是不甚明了,一则需要实际应用去锤炼,二来就是不懂观察者模式. 正文 观察者模式又叫发布-订阅模式,定义了一种一对多的依赖关系,让多个观察者对象同时监听某一主题对 ...
- 设计模式学习之观察者模式(Observer,行为型模式)(7)
1.观察者模式又叫做发布-订阅模式. 2.观察者模式定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象.这个主题对象在状态发生变化时,会通知所有观察者对象,使它们能够自动更新自己. 3 ...
- php设计模式学习之观察者模式
什么都不说,先看代码: interface userOperateImpl { public function operate($username); } class userLoginLog imp ...
- Java-马士兵设计模式学习笔记-观察者模式-模拟Awt Button
一.概述 Java 的Awt是 Observer模式,现用Java自己模拟awt中Button的运行机制 二.代码 1.Test.java import java.text.DateFormat; i ...
- Java-马士兵设计模式学习笔记-观察者模式-AWT简单例子
1.AWT简单例子 TestFrame.java import java.awt.Button; import java.awt.Frame; import java.awt.event.Action ...
- Java-马士兵设计模式学习笔记-观察者模式-读取properties文件改成单例模式
一.概述 1.目标:读取properties文件改成单例模式 二.代码 1.Test.java class WakenUpEvent{ private long time; private Strin ...
- Java-马士兵设计模式学习笔记-观察者模式-OOD 封装Listener
一.概述 childe类中的是关联监听者dad的,若要再增加监听者,会很不方便,且要修改代码.好的方法是封装监听者类,用addListener()方法动态添加监听者 二.代码 1.Test.java ...
- Java-马士兵设计模式学习笔记-观察者模式-OOD 封装event
把小孩醒来时的具体情况封装成事件类 Test.java class WakenUpEvent{ private long time; private String location; private ...
随机推荐
- 中了J.Fla的毒!
中了J.Fla的毒!中了J.Fla的毒!中了J.Fla的毒!中了J.Fla的毒!中了J.Fla的毒!中了J.Fla的毒!中了J.Fla的毒!中了J.Fla的毒!中了J.Fla的毒!中了J.Fla的毒! ...
- 8.中断按键驱动程序之poll机制
本节继续在上一节中断按键程序里改进,添加poll机制. 那么我们为什么还需要poll机制呢.之前的测试程序是这样: ) { read(fd, &key_val, ); printf(" ...
- makefile初步制作,arm-linux- (gcc/ld/objcopy/objdump)详解
在linux中输入vi Makefile 来实现创建Makefile文件 注意:命令行前必须加TAB键 例如:将两个文件led.c和crt0.S汇编文件,制作一个Makefile文件 led.bin ...
- Java线程:线程栈模型
要理解线程调度的原理,以及线程执行过程,必须理解线程栈模型. 线程栈是指某时刻时内存中线程调度的栈信息,当前调用的方法总是位于栈顶.线程栈的内容是随着程序的运行动态变化的,因此研究线程栈必须选择一个运 ...
- ★电车难题的n个坑爹变种
哲学家都不会做的电车难题变异 此题会答清华北大 "电车难题(Trolley Problem)"是伦理学领域最为知名的思想实验之一,其内容大致是: 一个疯子把五个无辜的人绑在电车轨道 ...
- 集美大学网络1413第十次作业成绩(团队六) -- 展示博客(Alpha版本)
题目 团队作业6--展示博客(Alpha版本) 团队作业6成绩 团队/分值 简介& 项目地址 项目目标 (典型用户. 功能描述. 预期用户数量) 如何满足 用户需求 已完成目标 团队分工 团 ...
- JAVA基础第三组(5道题)
11 [程序11] 题目:有1.2.3.4个数字,能组成多少个互不相同且无重复数字的三位数?都是多少? 1.程序分析:可填在百位.十位.个位的数字都是1.2.3.4.组成所有的排列后再去 掉不满足条件 ...
- 团队作业7——Alpha冲刺之事后诸葛亮(宣告项目失败团队解散)
一.项目进度 1.4月5日,团队组建.满怀希望的能做好这个项目 2.4月12日,需求分析. 3.4月21日,需求改进,出现协作问题,没有做好. 4.做项目,学习新的知识,继续做项目,但是能力有限,团队 ...
- 201521123053 <<Java基本语法与类库>>第二周
1.本周学习总结 这是第二周学习了,还是感觉吃力,慢慢来吧 .知识总是一点一点进到脑子里,虽然头小,但总会进来不是! 知识储备: A.整形变量按照长度分为byte.short.int.long. ...
- JAVA课程设计——“小羊吃蓝莓”小游戏
JAVA课程设计--"小羊吃蓝莓"小游戏 1. 团队课程设计博客链接 http://www.cnblogs.com/HXY071/p/7061216.html 2. 个人负责模块或 ...