最近在学习设计模式,用的是 《Head First 设计模式》这本书。感觉这本书写的还是很不错的,深入浅出的介绍了各种常用的设计模式。唯一有点不方便的地方是这本书的例子全都是用的 Java 来实现的。而我主要是用 C++。所以就动手将书上的代码用 C++ 来实现了一遍。

观察者模式

首先是三个接口的代码:

//observer.h
#ifndef OBSERVER_H
#define OBSERVER_H

class Observer
{
public:
    Observer() {}
    virtual void update(double temp, double humidity, double pressure) = 0;
};
#endif // OBSERVER_H
//Subject.h
#ifndef SUBJECT_H
#define SUBJECT_H

#include "observer.h"
#include <QList>
class Subject
{
public:
    Subject();
    void registerObserver(Observer *o);
    void removeObserver(Observer *o);
    void notifyObservers();
private:
    QList<Observer *> m_list;
protected:
    double m_temp;
    double m_humidity;
    double m_pressure;
};

#endif // SUBJECT_H
#include "subject.h"

Subject::Subject()
{

}

void Subject::registerObserver(Observer *o)
{
    m_list.append(o);
}

void Subject::removeObserver(Observer *o)
{
    int i = m_list.indexOf(o);
    if( i >= 0 )
    {
        m_list.removeAt(i);
    }
}

void Subject::notifyObservers()
{
    foreach (Observer * o, m_list)
    {
        o->update(m_temp, m_humidity, m_pressure);
    }
}
#ifndef DISPLAYELEMENT_H
#define DISPLAYELEMENT_H

class DisplayElement
{
public:
    DisplayElement() {}
    virtual void display() = 0;
};

#endif // DISPLAYELEMENT_H

下面是 WeatherData 类:

#ifndef WEATHERDATA_H
#define WEATHERDATA_H

#include "subject.h"

class WeatherData : public Subject
{
public:
    WeatherData();
    void setMeasurements(double t, double h, double p);
    void measurementsChanged();
};

#endif // WEATHERDATA_H
#include "weatherdata.h"

WeatherData::WeatherData()
{

}

void WeatherData::setMeasurements(double t, double h, double p)
{
   m_temp = t;
   m_humidity = h;
   m_pressure = p;
   measurementsChanged();
}

void WeatherData::measurementsChanged()
{
    notifyObservers();
}

最后是四种观察者:

#include "observer.h"
#include "displayelement.h"

class CurrentConditionDisplay : public Observer, public DisplayElement
{
public:
    CurrentConditionDisplay();
    void display() override;
    void update(double temp, double humidity, double pressure) override;
private:
    double m_temp;
    double m_humidity;
    double m_pressure;
};

class ForecastDisplay: public Observer, public DisplayElement
{
public:
    ForecastDisplay();
    void display() override;
    void update(double temp, double humidity, double pressure) override;
private:
    double currentPressure = 29.92f;
    double lastPressure;
};

class HeatIndexDisplay : public Observer, public DisplayElement
{
public:
    HeatIndexDisplay();
    void display() override;
    void update(double temp, double humidity, double pressure) override;
private:
    double m_heatIndex;
    double computeHeatIndex(double t, double rh);
};

class StatisticsDisplay: public Observer, public DisplayElement
{
public:
    StatisticsDisplay();
    void display() override;
    void update(double temp, double humidity, double pressure) override;
private:
    int m_numReadings;
    double m_tempSum;
    double m_maxTemp;
    double m_minTemp;
};
#include "display.h"
#include <iostream>

using std::cout;
using std::endl;

CurrentConditionDisplay::CurrentConditionDisplay()
{

}

void CurrentConditionDisplay::display()
{
    cout << "Current conditions:"  << m_temp
         << " F degrees and " << m_humidity
         << "% humidity" << endl;
}

void CurrentConditionDisplay::update(double temp, double humidity, double pressure)
{
    m_temp = temp;
    m_humidity = humidity;
    m_pressure = pressure;
    display();
}

ForecastDisplay::ForecastDisplay()
{

}

void ForecastDisplay::display()
{
    cout << "Forecast: ";
    if (currentPressure > lastPressure)
    {
        cout << "Improving weather on the way!";
    }
    else if (currentPressure == lastPressure)
    {
        cout << "More of the same";
    }
    else if (currentPressure < lastPressure)
    {
        cout << "Watch out for cooler, rainy weather";
    }
    cout << endl;
}

void ForecastDisplay::update(double temp, double humidity, double pressure)
{
    lastPressure = currentPressure;
    currentPressure = pressure;
    display();
}

HeatIndexDisplay::HeatIndexDisplay()
{

}

void HeatIndexDisplay::display()
{
    cout << "Heat index is " << m_heatIndex << endl;
}

void HeatIndexDisplay::update(double temp, double humidity, double pressure)
{
    (void) pressure;
    m_heatIndex = computeHeatIndex(temp, humidity);
    display();
}

double HeatIndexDisplay::computeHeatIndex(double t, double rh)
{
    double index = (double)((16.923 + (0.185212 * t) + (5.37941 * rh) - (0.100254 * t * rh)
        + (0.00941695 * (t * t)) + (0.00728898 * (rh * rh))
        + (0.000345372 * (t * t * rh)) - (0.000814971 * (t * rh * rh)) +
        (0.0000102102 * (t * t * rh * rh)) - (0.000038646 * (t * t * t)) + (0.0000291583 *
        (rh * rh * rh)) + (0.00000142721 * (t * t * t * rh)) +
        (0.000000197483 * (t * rh * rh * rh)) - (0.0000000218429 * (t * t * t * rh * rh)) +
        0.000000000843296 * (t * t * rh * rh * rh)) -
        (0.0000000000481975 * (t * t * t * rh * rh * rh)));
    return index;
}

StatisticsDisplay::StatisticsDisplay():
    m_tempSum(0.0),
      m_maxTemp(0),
      m_minTemp(200),
      m_numReadings(0)
{

}

void StatisticsDisplay::display()
{
    cout << "Avg/Max/Min temperature = " << m_tempSum / m_numReadings
         <<  "/" << m_maxTemp << "/" << m_minTemp << endl;
}

void StatisticsDisplay::update(double temp, double humidity, double pressure)
{
    (void) humidity;
    (void) pressure;

    m_tempSum += temp;
    m_numReadings++;

    if (temp > m_maxTemp)
    {
        m_maxTemp = temp;
    }

    if (temp < m_minTemp)
    {
        m_minTemp = temp;
    }

    display();
}

最后是测试代码:

int main(int argc, char *argv[])
{
    WeatherData weatherData;
    CurrentConditionDisplay currentDisplay;
    StatisticsDisplay staticDisplay;
    ForecastDisplay forcastDisplay;
    HeatIndexDisplay heatDisplay;
    weatherData.registerObserver(&currentDisplay);
    weatherData.registerObserver(&staticDisplay);
    weatherData.registerObserver(&heatDisplay);
    weatherData.registerObserver(&forcastDisplay);

    weatherData.setMeasurements(80, 65, 30.4);
    weatherData.setMeasurements(82, 70, 29.2f);
    weatherData.setMeasurements(78, 90, 29.2f);
}

《Head First 设计模式》例子的C++实现(2 观察者模式)的更多相关文章

  1. Java 设计模式系列(十五)观察者模式(Observer)

    Java 设计模式系列(十五)观察者模式(Observer) Java 设计模式系列目录(https://www.cnblogs.com/binarylei/p/10198698.html) Java ...

  2. php 设计模式 例子

    加载类:include("./Ren.class.php");include "./Ren.class.php"; require("./Ren.cl ...

  3. js设计模式中发布与订阅实现观察者模式例子

    <script> var pubsub = {}; (function(q) { var topics = {}; subuid = -1; q.publish = function(to ...

  4. 学C#之设计模式系列笔记(2)观察者模式

    一.借鉴说明 1.<Head First Design Patterns>(中文名<深入浅出设计模式>) 2.维基百科,观察者模式,https://zh.wikipedia.o ...

  5. Java设计模式(四)——再谈观察者模式

    在本系列的上一篇文章中,我们讨论了JDK对于观察者模式的一套实现.今天我们将要从另一个角度来探索Tomcat中是如何利用观察者模式加载各个组件.不过今天的任务不是解释Tomcat,所以我会单独把重点抽 ...

  6. JAVA设计模式详解(二)----------观察者模式

    有一个模式可以帮助你的对象知悉现况,不会错过该对象感兴趣的事,对象甚至在运行时可以决定是否要继续被通知,如果一个对象状态的改变需要通知很多对这个对象关注的一系列对象,就可以使用观察者模式 .观察者模式 ...

  7. [转] JavaScript设计模式之发布-订阅模式(观察者模式)-Part1

    <JavaScript设计模式与开发实践>读书笔记. 发布-订阅模式又叫观察者模式,它定义了对象之间的一种一对多的依赖关系.当一个对象的状态发生改变时,所有依赖它的对象都将得到通知. 例如 ...

  8. Java设计模式之(十二)——观察者模式

    1.什么是观察者模式? Define a one-to-many dependency between objects so that when one object changes state, a ...

  9. Java设计模式菜鸟系列(两)建模与观察者模式的实现

    转载请注明出处:http://blog.csdn.net/lhy_ycu/article/details/39755577 观察者(Observer)模式定义:在对象之间定义了一对多的依赖关系,这样一 ...

  10. 设计模式一:关于C++写观察者模式的一些收获

    先贴上部分代码: #include "stdafx.h" #include<iostream> #include<string> #include<v ...

随机推荐

  1. 三. Redis 主从复制

    特点 1. Master可以拥有多个Slave 2. 多个Slave除可以连接一个Master外,还可以连接多个Salve(避免Master挂掉不能同步,当Master挂掉,其中一个Slave会立即变 ...

  2. beta冲刺随笔集

    团队成员 郑西坤 031602542 (队长) 陈俊杰 031602504 陈顺兴 031602505 张胜男 031602540 廖钰萍 031602323 雷光游 031602319 吴志鸿 03 ...

  3. SQL Server 2012 读写分离设置 - AlsoIn

    原文转至:http://www.tuicool.com/articles/a6rmiam/ 引用: http://technet.microsoft.com/zh-cn/library/jj16176 ...

  4. C++ 获取当前正在执行的函数的相关信息

    (我的运行环境:win10x64+vs2015通过, 有的环境KUbuntu 8.04.1 x64 g++ 4.2.3也通过了)主要通过宏来实现:(注意,开头和结尾都是两个下划线) 1. __PRET ...

  5. 08-OpenLDAP主机控制策略

    OpenLDAP主机控制策略 阅读视图 参考 环境准备 openldap服务端配置 openldap客户端配置 客户端测试登录 故障处理 1. 参考 本文基本转载博客openldap主机访问控制(基于 ...

  6. 第六章 键盘(SYSMETS4)

    //SYSMETS.H -- System metrics display structure #include <Windows.h> #define NUMLINES ((int) ( ...

  7. git撤销本地提交但未push的记录

    ### git撤销本地提交但未push的记录 前言:有时候本地执行commit命令或者cherry-pick命令后发现提交了不需要提交的东西,就需要把已提交的commit记录撤销下来,简单做下记录 撤 ...

  8. MySQL基本简单操作02

    MySQL基本简单操作 先进入Mysql容器. [root@promote ~]# docker exec -it mysql /bin/bash root@30d60b852cf5:/# mysql ...

  9. Linux 小知识翻译 - 「Linux之父 Linus」

    作为新年的第一次,这次想简单介绍下Linus这个人.(这篇文章是作者新年初写的,所以有这么句话) Linux之父,同时也是现在linux内核开发最终决定的人物就是「Linus Torvalds」.「L ...

  10. Go学习笔记03-结构控制

    目录 条件语句 循环语句 条件语句 条件语句用 if 关键字来判定条件,如: func bounded(v int) int { if v > 100 { return 100 } else i ...