气象站的故事-观察者模式
前言
大家好!好久没有更新这个系列的文章了,这两个来月回家过了新年,公司搬了新家,就是这个系列的文章还没有更新,实在惭愧!同时再次真诚地感谢一直喜欢和支持这个系列文章的朋友们,因为你们的鼓励,我才有动力继续下去。可能因为这个系列每篇文章都比较长的原因,为了保证质量我总是字斟句酌,所以每次想动笔的时候都有点胆怯的感觉,但是还好每次只要写了开头我就会坚持把它写完的,还是万事开头难啊。
上篇【策略模式】得到了很多朋友的支持,倍感欣慰。这篇文章将延续以往的风格,故事来源同样取材于《Head First Design Patterns》,气象站的故事部分并非我原创,只是把原书的内容用更通俗易懂的方式展现给大家,特此声明一下,而这里更多的是作者对模式本身的理解和扩展的引申。关于HFDP,推荐大家去购买和阅读原版图书。
OK!我们这就开始!(提示:西红柿和鸡蛋都是好东西,请不要乱丢)
气象站的故事
现在我们要为一家气象站开发一套气象监控系统,按照客户的要求,这个监控系统必须可以实时跟踪当前的天气状况(温度、湿度、大气压力),并且可以在三种不同设备上显示出来(当前天气状况、天气统计、天气预测)。客户还希望这个系统可以对外提供一个API接口,以便任何开发者都可以开发自己的显示设备,然后无缝挂接到系统中,系统可以统一更新所有显示设备的数据。客户还会提供一个可以访问气象站的硬件设备的组件,如下图所示:
它提供了三个方法(get开头),可以分别取得实时的温度、湿度和大气压力,还有一个MeasurementsChanged()方法,当任何天气状况发生变化的时候,这个方法都会自动被触发,当前这个方法只是一个空函数,扩展的代码还需要我们自己去扩充。至于WeatherData是如何取得天气状况的,还有MeasurementsChanged()方法是如何被自动触发的这些事情都不需要我们去考虑,我们只管考虑如果做好跟显示设备有关的事情就好了。
OK!让我们来考虑一下这个系统的实现,先重新理一下思路:
1. 客户提供了获取实时的天气状况的方法。
2. MeasurementsChanged()方法会在天气状况变化时被自动调用。
3. 系统要实现三种显示模式,分别显示天气状况、天气统计和天气预测,而且这些显示的信息必须跟当前最新的天气状况实时同步。
4. 系统还必须支持在显示方式上的扩展性,而且使用者可以任意添加和移除不同的显示模式。
基于上面这些信息,我们大概都会想到可以象下面这样来实现这个系统:
//伪代码
public class WeatherData
{
//实例化显示设备(省略)
public void MeasurementsChanged()
{
float temp = getTemperature(); //取得温度
float humidity = getHumidity(); //取得湿度
float pressure = getPressure(); //取得气压
currentConditionsDisplay.update(temp, humidity, pressure); //同步显示当前天气状况
statisticsDisplay.update(temp, humidity, pressure); //同步显示天气统计信息
forecastDisplay.update(temp, humidity, pressure); //同步显示天气预报信息
}
}
因为客户已经给我们提供了实时的数据,还提供了数据更新时候的触发机制,那么我们要做的就是把最新的数据提供给不同的显示设备就OK了,上面的代码好象已经可以基本解决问题啦。哈哈!
真的就这么简单就搞定了吗?让我们用上一篇【策略模式】里学习到的原则来审视一下这个实现。首先,xxxDisplay 这几个对象都是具体的类实例,也就是说我们在这里违背了“面向接口编程,而不要面向实现编程。”的原则,这样实现会带来的问题是系统无法满足在不修改代码的情况下动态添加或移除不同的显示设备。换句话说,显示设备相关的部分是系统中最不稳定的部分,应该将其单独隔离开,也就是前面学过的另一个原则:“找到系统中变化的部分,将变化的部分同其它稳定的部分隔开。”那么我们到底该怎么办呢?呵呵,既然这篇文章是讲观察者模式的,当然要用它来结束战斗!下面我们先来认识一下观察者模式~
这就是观察者模式
我们还是先看一下官方的定义:
The Observer Pattern defines a one-to-many dependency between objects so that when one object changes state, all of its dependents are notified and updated automatically. (观察者模式定义了对象间的一种一对多依赖关系,使得每当一个对象改变状态,则所有依赖于它的对象都会得到通知并被自动更新)
咋样?这是超级经典的标准定义,如假抱换的!不懂?那再看看下面的类图吧~
Subject(被观察的对象接口)
l 规定ConcreteSubject的统一接口;
l 每个Subject可以有多个Observer;
ConcreteSubject(具体被观察对象)
l 维护对所有具体观察者的引用的列表;
l 状态发生变化时会发送通知给所有注册的观察者。
Observer(观察者接口)
l 规定ConcreteObserver的统一接口;
l 定义了一个update()方法,在被观察对象状态改变时会被调用。
ConcreteObserver(具体观察者)
l 维护一个对ConcreteSubject的引用;
l 特定状态与ConcreteSubject同步;
l 实现Observer接口,通过update()方法接收ConcreteSubject的通知。
怎么样,现在总该有点感觉了吧?下面还有一个顺序图,再体会体会~
呵呵!还没想明白,为什么官方的东西总是看不懂,看来是没当官的命啦!其实观察者模式十分简单,现实生活中的例子更是随处可见,就比如看电视:某个观众就是一个标准的ConcreteObserver(具体观察者,都符合统一的Observer接口,即都要通过电视收看节目的观众),电视节目就是Subject(被观察对象接口,这里体现为无线电视信号)了,不同的频道的节目是不同的ConcreteSubject(不同频道有不同的节目),观众可以自由决定看电视(registerObserver)或不看电视(removeObserver),而电视节目的变化也会在自动更新(notifyObservers)所有观众的收看内容。怎么样?这回明白了吧!
另外观察者模式也叫发布-订阅模式(Publishers + Subscribers = Observer Pattern),跟看电视一样,订阅报纸也是一个很直观的例子,有人发布(Publish = Subject)报纸,有人订阅(Subscribe = Observer)报纸,订阅的人可以定期收到最新发布的报纸,订阅人也可以随时退订。
现在大家应该对观察者模式基本都了解了,我们来用这个模式来解决气象站哪个问题。就气象站问题的应用场景来说,WeatherData可以作为ConcreteSubject来看待,而不同的显示设备则可以作为ConcreteObserver来看待,也就是说显示设备观察WeatherData对象,如果WeatherData对象有任何状态变化,则立刻更新显示设备的数据信息。这么说似乎很靠谱了,下面我们再来具体实现一下吧,先从整体结构开始,如下类图:
跟前面说的实现方式完全一样,只是这里为所有显示设备又定义了一个统一的接口,这个接口里定义了一个display()方法,也就是说未来所有实现Observer和DisplayElement接口的对象应该都可以作为气象监控系统的终端显示设备,不同用户可以在display()方法里任意自定义自己的显示模式。因为为了防止混乱,图4中只画了一个具体显示设备对象,即CurrentConditionsDisplay,跟它同级别的还有StatisticsDisplay和ForcastDisplay,它们在结构上完全相同。下面我们通过具体的代码再进一步理解一下基于观察者模式的气象监控系统的实现。
ISubject:
1
using System;
2
3
namespace DesignPatterns.Observer.WeatherData
4

{
5
public interface ISubject
6
{
7
void RegisterObserver(IObserver o);
8
void RemoveObserver(IObserver o);
9
void NotifyObserver();
10
}
11
}
12
关于这段代码,似乎没什么好说的了,因为上面已经反复说了很多啦。
IObserver:
1
using System;
2
3
namespace DesignPatterns.Observer.WeatherData
4

{
5
public interface IObserver
6
{
7
void Update(float temperature, float humidity, float pressure);
8
}
9
}
10
这里我们给update()方法定义了三个对应不同气象数据的参数。
IDisplayElement:
1
using System;
2
3
namespace DesignPatterns.Observer.WeatherData
4

{
5
public interface IDisplayElement
6
{
7
object Display();
8
}
9
}
10
这个类也是超级简单,没什么可解释的。
WeatherData:
1
using System;
2
using System.Collections;
3
4
namespace DesignPatterns.Observer.WeatherData
5

{
6
public class WeatherData : ISubject
7
{
8
private ArrayList observers;
9
private float temperature;
10
private float humidity;
11
private float pressure;
12
13
public WeatherData()
14
{
15
observers = new ArrayList();
16
}
17
18
ISubject Members#region ISubject Members
19
20
public void RegisterObserver(IObserver o)
21
{
22
observers.Add(o);
23
}
24
25
public void RemoveObserver(IObserver o)
26
{
27
int i = observers.IndexOf(o);
28
if(i >= 0)
29
{
30
observers.Remove(o);
31
}
32
}
33
34
public void NotifyObserver()
35
{
36
foreach(IObserver observer in observers)
37
{
38
observer.Update(temperature,humidity,pressure);
39
}
40
}
41
42
#endregion
43
44
public void MeasurementsChanged()
45
{
46
NotifyObserver();
47
}
48
49
public void SetMeasurements(float temperature, float humidity,
50
float pressure)
51
{
52
this.temperature = temperature;
53
this.humidity = humidity;
54
this.pressure = pressure;
55
MeasurementsChanged();
56
}
57
}
58
}
59
这个类是ISubject的具体实现,内部使用ArrayList来记录所有注册的观察者,SetMeasurements() 方法是用来模拟前面提到的在天气状况改变的时候自动触发MeasurementsChanged()方法的机制。
CurrentConditionsDisplay:
1
using System;
2
3
namespace DesignPatterns.Observer.WeatherData
4

{
5
public class CurrentConditionsDisplay : IObserver, IDisplayElement
6
{
7
private float temperature;
8
private float humidity;
9
private float pressure;
10
private ISubject weatherData;
11
12
public CurrentConditionsDisplay(ISubject weatherData)
13
{
14
this.weatherData = weatherData;
15
weatherData.RegisterObserver(this);
16
}
17
18
IObserver Members#region IObserver Members
19
20
public void Update(float temperature, float humidity, float pressure)
21
{
22
this.temperature = temperature;
23
this.humidity = humidity;
24
this.pressure = pressure;
25
}
26
27
#endregion
28
29
IDisplayElement Members#region IDisplayElement Members
30
31
public object Display()
32
{
33
return "Current conditions: " + temperature +
34
"F degrees and " + humidity + "% humidity";
35
}
36
37
#endregion
38
}
39
}
40
这个类是IObserver和IDisplayElement的具体实现,代表显示当前天气状况的具体显示设备对象,其内部维护了一个ISubject类型的变量,该变量在CurrentConditionsDisplay的构造函数中被初始化,同时调用ISubject.registerObserver()方法,实现订阅ISubject。
StatisticsDisplay和ForcastDisplay:
1
using System;
2
using System.Text;
3
4
namespace DesignPatterns.Observer.WeatherData
5

{
6
public class StatisticsDisplay : IObserver, IDisplayElement
7
{
8
Members#region Members
9
private float maxTemp = 0.0f;
10
private float minTemp = 200;
11
private float temperatureSum = 0.0f;
12
private int numReadings = 0;
13
private ISubject weatherData;
14
#endregion//Members
15
16
NumberOfReadings Property#region NumberOfReadings Property
17
public int NumberOfReadings
18
{
19
get
20
{
21
return numReadings;
22
}
23
}
24
#endregion//NumberOfReadings Property
25
26
Constructor#region Constructor
27
public StatisticsDisplay(ISubject weatherData)
28
{
29
this.weatherData = weatherData;
30
weatherData.RegisterObserver(this);
31
}
32
#endregion//Constructor
33
34
IObserver Members#region IObserver Members
35
36
public void Update(float temperature, float humidity, float pressure)
37
{
38
temperatureSum += temperature;
39
numReadings++;
40
41
if (temperature > maxTemp)
42
{
43
maxTemp = temperature;
44
}
45
46
if (temperature < minTemp)