观察者模式定义:
定义了对象之间的一对多依赖,这样一来,当一个对象改变状态时,它的所有依赖者都会受到通知并自动更新。
观察者模式又叫做发布-订阅(Publish/Subscribe)模式、模型-视图(Model/View)模式、源-监听器(Source/Listener)模式或从属者(Dependents)模式。
观察者模式定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象。这个主题对象在状态上发生变化时,会通知所有观察者对象,使它们能够自动更新自己
类图
图一 观察者模式类图
可以看出,在这个观察者模式的实现里有下面这些角色:
- 抽象主题(Subject)角色:主题角色把所有对观察考对象的引用保存在一个聚集里,每个主题都可以有任何数量的观察者。抽象主题提供一个接口,可以增加和删除观察者对象,主题角色又叫做抽象被观察者(Observable)角色,一般用一个抽象类或者一个接口实现。
- 抽象观察者(Observer)角色:为所有的具体观察者定义一个接口,在得到主题的通知时更新自己。这个接口叫做更新接口。抽象观察者角色一般用一个抽象类或者一个接口实现。在这个示意性的实现中,更新接口只包含一个方法(即Update()方法),这个方法叫做更新方法。
- 具体主题(ConcreteSubject)角色:将有关状态存入具体现察者对象;在具体主题的内部状态改变时,给所有登记过的观察者发出通知。具体主题角色又叫做具体被观察者角色(Concrete Observable)。具体主题角色通常用一个具体子类实现。
- 具体观察者(ConcreteObserver)角色:存储与主题的状态自恰的状态。具体现察者角色实现抽象观察者角色所要求的更新接口,以便使本身的状态与主题的状态相协调。如果需要,具体现察者角色可以保存一个指向具体主题对象的引用。具体观察者角色通常用一个具体子类实现。
观察者模式提供了一种对象设计,让主题和观察者之间松耦合。
当两个对象之间松耦合,他们依然可以交互,但是不太清楚彼此的细节。
关于观察者的一切,主题只知道观察者实现了某个接口(Observer)。主题不需要知道观察者的具体类是谁、做了些什么或其他任何细节。
任何时候我们可以增加新的观察者。因为主题唯一依赖的东西是一个实现Observer接口的对象列表。在有新的类型的观察者出现时,主题的代码不需要修改。所要做的就是在新的类里实现此观察者接口,然后注册为观察者即可。
改变主题或观察者其中一方,并不会影响另一方。因为两者是松耦合的。
气象站实例C++编程实现
(来源于《Head First设计模式》):
其类图:
代码:
//观察者模式C++编程实现
//author:江南烟雨
//E-Mail:[email protected]
#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;
static int setConditionTimes = 0;//记录设置的总次数
//观察者(布告板)抽象接口
class Observer{
public:
virtual void update(float temp,float humidity,float pressure) = 0;
};
//主题抽象接口
class Subject{
public:
virtual void registerObserver(Observer *o) = 0;//注册观察者
virtual void removeObserver(Observer *o) = 0;//删除观察者
virtual void notifyObserver() = 0;//当主题状态改变时,通知所有观察者
};
//布告板显示
class DisplayElement{
public:
virtual void display()= 0;
};
//WeatherData具体类,继承自主题抽象类
class WeatherData:public Subject{
private:
vector<Observer *> observers;
float temperature;
float humility;
float pressure;
public:
WeatherData(){}
~WeatherData(){
vector<Observer*>::iterator pos = observers.begin();
for (;pos != observers.end();++pos)
{
delete (*pos);
}
}
void registerObserver(Observer *o){
observers.push_back(o);
}
void removeObserver(Observer *o){
vector<Observer*>::iterator pos = find(observers.begin(),
observers.end(),o);
if (pos != observers.end())
{
observers.erase(pos);
}
}
void notifyObserver()
{
vector<Observer*>::iterator pos = observers.begin();
for (;pos != observers.end();++pos)
{
(*pos)->update(temperature,humility,pressure);
}
}
//当从气象站得到更新观测值时,通知观察者
void messurementsChanged(){
notifyObserver();
}
//设置气象站观测值
void setMessurements(float temperature,float humility,float pressure){
++setConditionTimes;
this->temperature = temperature;
this->humility = humility;
this->pressure = pressure;
messurementsChanged();
}
};
//布告板具体类:目前状况
class CurrentConditionDisplay:public Observer,public DisplayElement{
private:
float temperature;
float humility;
Subject *weatherData;
public:
CurrentConditionDisplay(Subject *weatherData){
this->weatherData = weatherData;
weatherData->registerObserver(this);
}
~CurrentConditionDisplay(){}
void update(float temp,float humi,float pressure){
this->temperature = temp;
this->humility = humi;
display();
}
void display(){
cout << "Current conditions: " << temperature << "F degrees and " <<
humility << "% humility" << endl;
}
};
//布告板具体类:气象统计类
class WeatherStatisticsDisplay:public Observer,public DisplayElement{
private:
float avgTemperatuere,maxTemperature,minTemperature;
Subject *weatherData;
public:
WeatherStatisticsDisplay(Subject *weatherData){
this->weatherData = weatherData;
weatherData->registerObserver(this);
}
~WeatherStatisticsDisplay(){}
void update(float temp,float humi,float pressure){
if (1 == setConditionTimes)
{
this->avgTemperatuere = temp;
this->maxTemperature = temp;
this->minTemperature = temp;
}
else{
this->avgTemperatuere = (this->avgTemperatuere * (setConditionTimes - 1) + temp) / setConditionTimes;
this->maxTemperature = this->maxTemperature > temp ? this->maxTemperature : temp;
this->minTemperature = this->minTemperature < temp ? this->minTemperature : temp;
}
display();
}
void display(){
cout << "Avg/Max/Min: temperature " << avgTemperatuere << "/" <<
maxTemperature << "/" <<minTemperature << endl;
}
};
int main()
{
WeatherData *weatherData = new WeatherData();
CurrentConditionDisplay *currentConditionDisplay = new CurrentConditionDisplay(weatherData);
WeatherStatisticsDisplay *weatherStatisticsDisplay = new WeatherStatisticsDisplay(weatherData);
weatherData->setMessurements(80,65,30.4f);
weatherData->setMessurements(82,70,29.2f);
weatherData->setMessurements(78,90,29.2f);
weatherData->removeObserver(weatherStatisticsDisplay);
cout << "after remove observer WeatherStatisticsDisplay:" << endl;
weatherData->setMessurements(80,65,30.4f);
weatherData->setMessurements(82,70,29.2f);
weatherData->setMessurements(78,90,29.2f);
return 0;
}
运行结果(VS2008 + Win7):