最近在学习设计模式,用的是 《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. (网页)js每隔5分钟执行一次ajax请求的实现方法(转)

    转自脚本之家: 一个页面好像只能有一个 window.onload=function(){},所以要有多个事件,这样写就好了 window.onload=function(){ //假设这里每个五分钟 ...

  2. 第四周读书笔记——读《我是一只IT小小鸟》有感

             读<我是一只IT小小鸟>有感 这是邓老师倾力推荐的一本书.这本书的标题化用了我们耳熟能详的歌词,算是较有新意吧.更重点在于,这本书的作者不是哪一位大牛,而是一群刚刚走出校 ...

  3. SQL Server 2005详细安装过程及配置

    说明:个人感觉SQL Server 2005是目前所有的SQL Server版本当中最好用的一个版本了,原因就是这个版本比起其它版本来说要安装简单,操作简便,只可惜这个版本只能在Windows7或者低 ...

  4. python第六十五天--python操作mysql

    pymysql模块对mysql进行 import pymysql # 创建连接 conn = pymysql.connect(host='127.0.0.1', port=3306, user='ro ...

  5. C#与Java AES 加密解密

    参考文档:https://www.cnblogs.com/xbzhu/p/7064642.html 前几天对接Java接口,需要C#加密参数,Java解密.奈何网上找了一堆大同小异的加解密方法都跟Ja ...

  6. Centos7.5.1804永久生效修改主机名

    原来主机名 [root@node1 ~]# 查看Centos的版本: [root@node1 ~]# cat /etc/redhat-release CentOS Linux release (Cor ...

  7. VB6 对象库未注册问题

    以下是个人使用VB6出现对象库未注册问题的解决方法.已成功! 一.注册ocx文件 mscomctl.ocx文件放进路径C:\Windows\System32(64是此路径,由于本人是64位系统32位未 ...

  8. 01LaTeX学习系列之---TeX的介绍与认识

    目录 01TeX的介绍与认识 目录 前言 (一)TeX 的宣传 TeX - Beauty and Fun 1. TeX 是什么? 2. TeX 是哪家公司生产的? 3. 我们今天主角的名字怎么念啊?& ...

  9. 从源码的角度分析List与Set的区别

    很多时候我们在讨论List与Set的异同点时都在说: 1.List.Set都实现了Collection接口 2.List是有序的,可以存储重复的元素,允许存入null 3.Set是无序的,不允许存储重 ...

  10. 课外知识----easy BCD

    可以用来作系统之间的切换 使用方法略