观察者模式
注:
1. 观察者模式: 又叫---发布---订阅模式
定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象。这个主题对象在状态发生改变时,会通知所有观察者对象,使得它们能够自动更新自己。
2. 观察者模式解析:
特点:观察者模式所做的工作其实就是在解除耦合,让耦合的双方都依赖于抽象,而不是依赖于具体。从而使得各自的变化不会影响另一边的变化。
什么时候使用?------当一个对象的改变,需要同时改变其他对象的时候,而且不知道有多少对象待改变,考虑使用观察者模式。
3. 观察者模式的例子:
Subject类,抽象通知类,一般用一个抽象类或者一个接口实现。它把所有对观察者对象的引用保存在一个聚集里,每个主题都可以有任意数量的观察者。抽象主题可以提供一个接口,可以增加和删除观察者对象。
Observer类,抽象观察者,为所有的具体观察者定义一个接口,在得到主题的通知时更新自己。这个接口叫更新接口。抽象观察者一般用一个抽象类或者一个接口实现。更新接口通常包含一个update()方法,叫更新方法。
4. 代码实例:
#include <iostream>
#include <string>
#include <list>
using namespace std;
class Subject;
class Observer//抽象观察者
{
protected:
string name;
Subject *sub;
public:
Observer(string name,Subject *sub)
{
this->name=name;
this->sub=sub;
}
virtual void update()=0;
};
class StockObserver:public Observer
{
public:
StockObserver(string name,Subject *sub):Observer(name,sub)
{
}
void update();
};
class NBAObserver:public Observer
{
public:
NBAObserver(string name,Subject *sub):Observer(name,sub)
{
}
void update();
};
class Subject//抽象的通知者接口
{
protected:
list<Observer*> observers;
public:
string action;
virtual void attach(Observer*)=0;
virtual void detach(Observer*)=0;
virtual void notify()=0;
};
class Secretary:public Subject
{
void attach(Observer *observer)
{
observers.push_back(observer);
}
void detach(Observer *observer)
{
list<Observer *>::iterator iter=observers.begin();
while(iter!=observers.end())
{
if((*iter)==observer)
{
observers.erase(iter);
}
++iter;
}
}
void notify()
{
list<Observer *>::iterator iter=observers.begin();
while(iter!=observers.end())
{
(*iter)->update();
++iter;
}
}
};
void StockObserver::update()
{
cout<<name<<" 收到消息:"<<sub->action<<endl;
if(sub->action=="梁所长来了!")
{
cout<<"我马上关闭股票,装做很认真工作的样子!"<<endl;
}
}
void NBAObserver::update()
{
cout<<name<<" 收到消息:"<<sub->action<<endl;
if(sub->action=="梁所长来了!")
{
cout<<"我马上关闭NBA,装做很认真工作的样子!"<<endl;
}
}
int main()
{
Subject *dwq=new Secretary();
Observer *xs=new NBAObserver("xiaoshuai",dwq);
Observer *zy=new NBAObserver("zouyue",dwq);
Observer *lm=new StockObserver("limin",dwq);
dwq->attach(xs);
dwq->attach(zy);
dwq->attach(lm);
dwq->action="去吃饭了!";
dwq->notify();
cout<<endl;
dwq->action="梁所长来了!";
dwq->notify();
return 0;
}