Observer模式要解决的问题为:建立一个一(Subject)对多(Observer)的依赖关系,并且做到当“一”变化的时候,依赖这个“一”的多也能够同步改变
观察者模式(有时又被称为发布/订阅模式)是软件设计模式的一种。在此种模式中,一个目标对象管理所有相依于它的观察者对象,并且在它本身的状态改变时主动发出通知。这通常透过呼叫各观察者所提供的方法来实现。此种模式通常被用来实作事件处理系统。
如下例子,当数据改变的时候,通知重新绘制所有图表,此数据是通过方法传过来的。
observer还可以保留subject的指针,这样就不需要通过方法参数而是通过subject指针来获取数据,如下图:
#include <iostream>
#include <list>
#include <string>
#include <numeric>
#include <vector>
typedef std::string Data;
class IDiagramShow
{
public:
IDiagramShow() {}
virtual ~IDiagramShow() {}
virtual void update(const Data &data) = 0;
};
class IDiagramData
{
public:
IDiagramData() {}
virtual ~IDiagramData() {}
virtual void add(IDiagramShow *p) = 0;
virtual void del(IDiagramShow *p) = 0;
virtual void notify() = 0;
};
class StockData : public IDiagramData
{
public:
void add(IDiagramShow *p)
{
observers_.push_back(p);
}
void del(IDiagramShow *p)
{
observers_.remove(p);
}
void notify()
{
for (std::list<IDiagramShow*>::iterator start=observers_.begin(), end=observers_.end();
start != end; ++start)
{
(*start)->update(data_);
}
}
void dataChange(const Data &data)
{
data_ = data;
}
private:
Data data_;
std::list<IDiagramShow*> observers_;
};
class ColumnDiagram : public IDiagramShow
{
public:
ColumnDiagram() {}
explicit ColumnDiagram(IDiagramData *p)
{
p->add(this);
}
void update(const Data &data)
{
std::cout << "column diagram:" << data << std::endl;
}
};
class TreeDiagram : public IDiagramShow
{
public:
TreeDiagram() {}
explicit TreeDiagram(IDiagramData *p)
{
p->add(this);
}
void update(const Data &data)
{
std::cout << "tree diagram:" << data << std::endl;
}
};
int _tmain(int argc, _TCHAR* argv[])
{
IDiagramShow *show1 = new ColumnDiagram(); // 观察者A
IDiagramShow *show2 = new TreeDiagram(); // 观察者B
StockData *data = new StockData();
data->add(show1);
data->add(show2);
data->dataChange("first...");
data->notify();
data->dataChange("second... ");
data->notify();
system("pause");
return 0;
}