最近在学习设计模式,用的是 《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. MVC与单元测试实践之健身网站(八)-统计分析

    ​统计分析模块与之前的内容相对独立,用于记录并跟踪各部位围度的变化.还需提供对所作计划的分析,辅助使计划更合理. 一 围度记录 这儿可以记录各项身体围度指标,现在包括体重在内身体上上下下基本全部提供了 ...

  2. (网页)js最新手机号码、电话号码正则表达式

    正则表达式(regular expression)是一个描述字符模式的对象.使用JavaScript正则表达式可以进行强大的模式匹配和文本检索与替换功能. 手机号码正则表达式验证. function ...

  3. (后端)Java新人入职——配置环境及安装开发工具(完全)

    转自csdn:执笔记忆的空白 很多新人对于进入新公司,相关工具的安装和环境变量的设定很苦恼.又苦于没有完整的配置开发环境的资料,我这里写一篇操作步骤的案例, 至少让你能把开发工具安装起来,并实用起来, ...

  4. 实现一个函数clone,可以对JavaScript中的5种主要的数据类型(包括Number、String、Object、Array、Boolean)进行值复制

    记录一下,方便以后复制粘贴 // 方法一: Object.prototype.clone = function() { var o = this.constructor === Array ? [] ...

  5. gitlab 和 github 配置 SSH Keys

    gitlab 文档上给了很好的配置的例子:https://gitlab.com/help/ssh/README#locating-an-existing-ssh-key-pair 针对mac 下的使用 ...

  6. SonarQube 配置 LDAP(AD域)

    安装插件 1.下载 LDAP Plugin 插件,地址:https://docs.sonarqube.org/display/SONARQUBE67/LDAP+Plugin2.将下载的插件,放到 SO ...

  7. January 17th, 2018 Week 03rd Wednesday

    Don't let go too soon, but don't hold on too long. 不要太快放手,也别紧握太久. It is inevitalbe to encounter with ...

  8. 5、爬虫系列之scrapy框架

    一 scrapy框架简介 1 介绍 (1) 什么是Scrapy? Scrapy是一个为了爬取网站数据,提取结构性数据而编写的应用框架,非常出名,非常强悍.所谓的框架就是一个已经被集成了各种功能(高性能 ...

  9. C# 响应微信发送的Token验证,文字、图文自动回复、请求客服对话.....

    代码如下,有需要的可以参考: using System; using System.Collections.Generic; using System.Linq; using System.Web; ...

  10. 1407: [Noi2002]Savage

    其实答案远不到1e6 所以可以枚举! 设答案是m 那\(i,j\)的相遇就可以表示成\(P_ix+C_i=P_jx+C_j+ym\) 移向就是\((P_i-P_j)x-ym=C_j-C_i\) 套扩展 ...