设计模式之观察者模式

查看目录请点人家 ˙ω˙

气象观测站

你的团队接到一个任务,建立一个能显示气象站数据的应用,可以添加或移除布告板,布告板默认有 目前状况,气象统计,天气预测。用户可以随时添加或移除布告板。提供一个WeatherData对象,可以从气象站获得温度,湿度,气压等信息。
大概长这样
1-5

错误的实现

WeatherData类长这样
1-6
直接再函数中通知每一个布告板

void WeatherData::measurementsChanged(){
	float temp = getTemperature();
	float humidity = getHumidity();
	float pressure = getPressure();
	//通知每一个布告板
	bgb1.updata(temp,humidity,pressure);
	bgb2.updata(temp,humidity,pressure);
	bgb3.updata(temp,humidity,pressure);
}

有哪些不妥

  1. 我们是针对实现编程,而不是针对接口。
  2. 对于每一个布告板,我们都要修改代码
  3. 我们无法动态的增加或删除布告板
    。。。

使用观察者模式

出版者 + 订阅者 = 观察者模式
1-7
设计好出题和观察者的抽象基类,让WeatherData继承主题,布告板继承观察者。实现方式有很多,一种是Subject维护一个Observer对象的数组

class Observer{
    virtual void update(float temp,float humidity,float pressure)=0;
};

class Subject{
public:
    virtual void registerObserver(Observer * po)=0;
    virtual void removeObserver(Observer * po)=0;
    virtual void notifyObservers()=0;
protected:
    vector<Observer *> oblist;
};

具体实现
WeatherData

class WeatherData:public Subject{
public:
    virtual void registerObserver(Observer * po);
    virtual void removeObserver(Observer * po);
    virtual void notifyObservers();
    void setdata(float t,float h,float p);
private:
    float temperature;
    float humidity;
    float pressure;

};
void WeatherData::registerObserver(Observer * po){
    oblist.push_back(po);
}
void WeatherData::removeObserver(Observer * po){
    int i;
    for(i=0;i<oblist.size();i++){
        if(oblist[i]==po){
            oblist.erase(oblist.begin()+i);
            break;
        }
    }
}
void WeatherData::notifyObservers(){
    for(int i=0;i<oblist.size();i++){
        oblist[i]->update(temperature,humidity,pressure);
    }
}
void WeatherData::setdata(float t,float h,float p){
    temperature=t;
    humidity=h;
    pressure=p;
    notifyObservers();
}

布告板

class Bgb:public Observer{//布告板
public:
    virtual void update(float temp,float humidity,float pressure);
    void display();//打印
private:
    float t,h,p;//存放接受数据
};
void Bgb::update(float temp,float humidity,float pressure){
    t=temp; h=humidity; p=pressure;
}
void Bgb::display(){
    cout<<"Bjb * is"<<(int *)(this) <<endl;
    cout<<"temperature is :"<<t<<" humidity is :"<<h<<" pressure is :"<<p<<endl;
}

完整测试代码

#include <iostream>
#include <vector>
using namespace std;

class Observer{
public:
    virtual void update(float temp,float humidity,float pressure)=0;
};

class Subject{
public:
    virtual void registerObserver(Observer * po)=0;
    virtual void removeObserver(Observer * po)=0;
    virtual void notifyObservers()=0;
protected:
    vector<Observer *> oblist;
};
//----------------------------------------------------------
class WeatherData:public Subject{
public:
    virtual void registerObserver(Observer * po);
    virtual void removeObserver(Observer * po);
    virtual void notifyObservers();
    void setdata(float t,float h,float p);
private:
    float temperature;
    float humidity;
    float pressure;

};
class Bgb:public Observer{//布告板
public:
    virtual void update(float temp,float humidity,float pressure);
    void display();//打印
private:
    float t,h,p;//存放接受数据
};
//----------------------------------------------------------
int main(){
    WeatherData WD; //一个主题
    Bgb bjb1,bjb2,bjb3; //三个观察者
    WD.registerObserver(&bjb1);//注册
    WD.registerObserver(&bjb2);
    WD.registerObserver(&bjb3);
    WD.setdata(1,2,3);
    bjb1.display();//打印每个观察者得到的信息
    bjb2.display();
    bjb3.display();
    cout<<endl;
    WD.removeObserver(&bjb3);//bjb3 取消订阅
    WD.setdata(8,9,10);
    bjb1.display();//再次打印
    bjb2.display();
    bjb3.display();
    //bjb3 的信息没有更新
    return 0;
}
void Bgb::update(float temp,float humidity,float pressure){
    t=temp; h=humidity; p=pressure;
}
void Bgb::display(){
    cout<<"Bjb * is"<<(int *)(this) <<endl;
    cout<<"temperature is :"<<t<<" humidity is :"<<h<<" pressure is :"<<p<<endl;
}
void WeatherData::registerObserver(Observer * po){
    oblist.push_back(po);
}
void WeatherData::removeObserver(Observer * po){
    int i;
    for(i=0;i<oblist.size();i++){
        if(oblist[i]==po){
            oblist.erase(oblist.begin()+i);
            break;
        }
    }
}
void WeatherData::notifyObservers(){
    for(int i=0;i<oblist.size();i++){
        oblist[i]->update(temperature,humidity,pressure);
    }
}
void WeatherData::setdata(float t,float h,float p){
    temperature=t;
    humidity=h;
    pressure=p;
    notifyObservers();
}

程序输出
1-8
首先3个布告板都订阅,更新数据 1 2 3
然后布告板3取消订阅
更新数据 8 9 10
只有bgb3没有收到

观察者模式

定义了对象之间一对多的依赖,这样一来,当一个对象改变状态时,他的所有依赖者都会收到通知并自动更新。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值