设计模式入门(二)观察者模式

news2025/1/18 1:54:53

设计模式入门

本系列所有内容参考自《HeadFirst设计模式》。因为书中的代码是采用java语言写的,博主这里用C++语言改写。
这里采用讲故事的方式进行讲解。若有错误之处,非常欢迎大家指导。
设计模式:模式不是代码,而针对设计问题的通用解决方案,被认为是历经验证的OO设计经验。设计模式告诉我们如何组织类和对象以解决某种问题。
如果你输出一个helloworld都想使用设计模式的话,那可能真的就有问题了。

正文

提出问题

我们现在手头有一个气象检测应用。气象站接收湿度感应装置温度感应装置气压感应装置的数据,然后我们有一个WeatherData对象,它负责追踪来自气象站的数据,并更新布告板(显示目前天气状况给用户看)。
图片来自HeadFirst设计模式
如果我们要接手这个项目,我们的工作就是建立一个应用,利用WeatherData对象取得数据,并更新三个布告板:目前状况、气象统计和天气预报。三个布告板如下图所示:
在这里插入图片描述

现有的WeatherData类源码如下:

class WeatherData {
    float getTemperature();  //返回温度
    float getHumidity();     //返回湿度
    float getPressure();     //返回气压
    void measurementsChanged()
    {
        /*一旦气象测量更新,此方法会被调用*/
        //我们的代码加在这里
    }
};

我们的工作是实现measurementsChanged(),好让它更新目前状况、气象统计、天气预报的显示布告板。

我们目前知道的:WeatherData类有三个方法,可以取得三个测量值;当新的数据来临时,measurementsChanged()方法就会被调用(我们不在乎此方法是如何被调用的,我们只在乎它被调用了);我们需要实现三个使用天气数据的布告板,一旦WeatherData有新的测量,这些布告必须马上更新。

一个我们可能想到的measurementsChanged()实现如下:

class WeatherData {
	// 实例变量声明
    void measurementsChanged()
    {
    	// 获取最新的测量值
        float temp = getTemperature();
        float humidity = getHumidity();
        float pressure = getPressure();
        
        // 调用每个布告板更新显示
        currtenConditionsDisplay.update(temp, humidity, pressure);  // 目前状况布告板更新
        statisticsDisplay.update(temp, humidity, pressure);  // 气象统计布告板更新
        forecastDisplay.update(temp, humidity, pressure);   // 天气预报布告板更新
    }
};

但是这与一些软件设计原则发生了矛盾。上面代码中调用每个布告板更新显示函数是针对具体实现编程,会导致我们以后在增加或删除布告板时必须修改程序;三个接口都是update,传入的参数也是一样的,所以看起来更像是一个统一的接口。

那我们该如何解决这个问题呢?观察者模式可以帮助我们很好地解决这个问题。

观察者模式

一个很简单的例子就是杂志订阅
假设我们订阅了一款杂志,每当这款杂志更新时,它都会给我们送一份。这就是观察者模式,杂志相当于“主题”,我们相当于“观察者”,当主题发生改变时,就是通知“观察者”。这里要注意的一点是:主题来增加或删除观察者。 还是杂志订阅这个问题,我们想订阅杂志的时候,杂志出版社便会将我们加到它们的订阅名单里,我们不想订阅杂志时,杂志出版社便会将我们从订阅名单里删除。

观察者模式:观察者模式定义了对象之间的一对多依赖(“一个主题”对“多个观察者”),这样一来,当一个对象改变状态时,它的所有依赖者(因为主题是真正拥有数据的人,观察者是主题的依赖者)都会收到通知并自动更新。

实现代码如下:

#include<iostream>
#include<vector>

using namespace std;

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

class Subject {  // 抽象主题
    virtual void registerObserver(Observer *o)=0;
    virtual void removeObserver(Observer *o)=0;
    virtual void notifyObserver()=0;
};

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

class WeatherData : public Subject  // 具象主题
{
private:
    vector<Observer*> observers;
    float temperature;
    float humidity;
    float pressure;
public:
    void registerObserver(Observer *o)  // 注册观察者
    {
        observers.push_back(o);
    }

    void removeObserver(Observer *o)   // 取消观察者
    {
        auto it = std::find(observers.begin(), observers.end(), o);

        if (it != observers.end())
        {
            int index = std::distance(observers.begin(), it);
            cout << "索引是:" << index << endl;;
            observers.erase(observers.begin() + index);
            cout << "成功删除元素" << endl;
        }
        else
        {
            cout << "未找到元素" << endl;
        }
    }

    void notifyObserver()  // 通知观察者
    {
        for (int i = 0; i < observers.size(); i++)
        {
            Observer *observer = observers[i];
            observer->update(temperature, humidity, pressure);
        }
    }
    void measurementsChanged()
    {
        notifyObserver();  // 通知观察者
    }

    void setMeasurements(float temperature, float humidity, float pressure)
    {
        this->temperature = temperature; 
        this->humidity = humidity;
        this->pressure = pressure;
        measurementsChanged();
    }
};

class StatisticsDisplay : public Observer, public DisplayElement  // 观察者
{
private:
    float temperature;
    float humidity;
    WeatherData *weatherData;
public:
    StatisticsDisplay(WeatherData *weather)
    {
        weatherData = weather;
        weatherData->registerObserver(this);  //主题注册观察者
    }

    void remove()
    {
        weatherData->removeObserver(this); // 主题取消观察者
    }

    void update(float temperature, float humidity, float pressure)
    {
        this->temperature = temperature;
        this->humidity = humidity;
        display();
    }
    void display()
    {
        cout << "statisticsDisplay: " << temperature << "F degress and " << humidity << "% humidity" << endl;
    }
};

class ForecastDisplay : public Observer, public DisplayElement  // 观察者
{
private:
    float temperature;
    float humidity;
    WeatherData *weatherData;
public:
    ForecastDisplay(WeatherData *weather)
    {
        weatherData = weather;
        weatherData->registerObserver(this);  //主题注册观察者
    }

    void remove()
    {
        weatherData->removeObserver(this); // 主题取消观察者
    }

    void update(float temperature, float humidity, float pressure)
    {
        this->temperature = temperature;
        this->humidity = humidity;
        display();
    }
    void display()
    {
        cout << "ForecastDisplay: " << temperature << "F degress and " << humidity << "% humidity" << endl;
    }
};

class CurrentConditionsDisplay : public Observer, public DisplayElement  // 观察者
{
private:
    float temperature;
    float humidity;
    WeatherData *weatherData;
public:
    CurrentConditionsDisplay(WeatherData *weather)
    {
        weatherData = weather;
        weatherData->registerObserver(this);  //主题注册观察者
    }

    void remove()
    {
        weatherData->removeObserver(this); // 主题取消观察者
    }

    void update(float temperature, float humidity, float pressure)
    {
        this->temperature = temperature;
        this->humidity = humidity;
        display();
    }
    void display()
    {
        cout << "CurrentConditionsDisplay: " << temperature << "F degress and " << humidity << "% humidity" << endl;
    }
};
int main()
{
    WeatherData *weatherData = new WeatherData; // 定义一个主题对象即可
    CurrentConditionsDisplay currentDisplay(weatherData);  // 第一个观察者
    StatisticsDisplay statisDisplay(weatherData);   // 第二个观察者
    ForecastDisplay foreDisplay(weatherData);   // 第三个观察者
    weatherData->setMeasurements(80, 65, 30.4);   // 主题信息发生变更
    
    currentDisplay.remove();   // 该观察者取消对主题的订阅

    weatherData->setMeasurements(40, 25, 15.4);

    foreDisplay.remove();   // 该观察者取消对主题的订阅

    weatherData->setMeasurements(15.5, 26, 34);
    return 0;
}

以上就是使用C++实现观察者模式的全部代码。

设计原则

  1. 找出程序中会变化的方面,然后将其和固定不变的方面相分离。
    在观察中模式中,会改变的是主题的状态,以及观察者的数目和类型。用这个模式,你可以改变依赖于主题状态的对象,却不必改变主题。
  2. 针对接口编程,不针对实现编程。
    主题与观察者都是用接口:观察者利用主题的接口向主题注册,而主题利用观察者接口通知观察者。这样可以让两者之间运作正常,又同时具有松耦合的优点。

观察者模式较为重要,在很多软件框架和软件设计中都可以看到它的身影,所以大家可以根据代码仔细体会它的思想。工作的那几个月在公司的软件里看到过观察者模式,但是没有自己动手实现,只是明白它的意思。今天自己动手实现了一下,感悟又深了一些。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/967513.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

A Mathematical Framework for Transformer Circuits—Part (1)

A Mathematical Framework for Transformer Circuits 前言Summary of ResultsREVERSE ENGINEERING RESULTSCONCEPTUAL TAKE-AWAYS Transformer OverviewModel SimplificationsHigh-Level ArchitectureVirtual Weights and the Residual Stream as a Communication ChannelVIRTU…

用树形dp+状压维护树上操作的计数问题:0902T3

发现操作数 k ≤ 6 k\le6 k≤6&#xff0c;可以考虑对操作进行状压。 然后找找性质&#xff0c;发现要么删掉一棵子树&#xff0c;要么进去该子树。可以视为每种操作有两种情况。 然后分讨一下当前该如何转移。 树形dp的顺序&#xff1a; 合并子树考虑当前往上的边的方向 …

自定义类型:结构体、枚举、联合

目录 结构体 结构体的基础知识 结构的声明 特殊的声明 结构体的自引用 结构体变量的定义和初始化 结构体内存对齐 修改默认对齐数 结构体传参 位段 什么是位段 位段的内存分配 位段的跨平台问题 位段的应用 枚举 枚举类型的定义 枚举的优点 联合体&#xff08;共…

Mysql创建用户并且给指定用户添加某个库的所有权限

注意&#xff1a; 运行以下命令首先运行的用户需要有以下操作权限才可以执行 创建用户 创建用户&#xff1a;命令中的’username’替换为您要创建的用户名&#xff0c;‘host’替换为用户的主机名或IP地址如果都可以访问则配置’%&#xff0c;password’替换为用户的密码 CRE…

WoW GM

当年黑翼被人黑G以后&#xff0c;后来我就自己开团&#xff0c;今天整理电脑还发现截图。。。。

AI绘画:StableDiffusion实操教程-斗罗大陆2-江楠楠-常服(附高清图下载)

前段时间我分享了StableDiffusion的非常完整的教程&#xff1a;“AI绘画&#xff1a;Stable Diffusion 终极宝典&#xff1a;从入门到精通 ” 尽管如此&#xff0c;还有读者反馈说&#xff0c;尽管已经成功安装&#xff0c;但生成的图片与我展示的结果相去甚远。真实感和质感之…

h5页面使用js注入企业微信sdk

let script document.createElement(script); //定义script标签 script.type text/javascript; script.src http://res.wx.qq.com/open/js/jweixin-1.2.0.js; document.getElementsByTagName(body)[0].appendChild(script); //将script标签插入到body下面

ISO/IEC标准组织介绍(三十七)

简介: CSDN博客专家,专注Android/Linux系统,分享多mic语音方案、音视频、编解码等技术,与大家一起成长! 优质专栏:Audio工程师进阶系列【原创干货持续更新中……】🚀 人生格言: 人生从来没有捷径,只有行动才是治疗恐惧和懒惰的唯一良药. 更多原创,欢迎关注:Android…

[论文笔记]SiameseNet

引言 这是Learning Text Similarity with Siamese Recurrent Networks的论文笔记。 论文标题意思是利用孪生循环神经网络学习文本相似性。 什么是孪生神经网络呢?满足以下两个条件即可: 输入是成对的网络结构和参数共享(即同一个网络)如下图所示: 看到这种图要知道可能代…

C语言:截断+整型提升+算数转换练习

详情关于整型提升、算数转换与截断见文章&#xff1a; 《C语言&#xff1a;整型提升》 《C语言&#xff1a;算数转换》 一、代码一 int main() { char a -1; signed char b -1; unsigned char c -1; printf("%d %d %d", a, b, c); return 0; } 求…

mp代码生成插件

mp代码生成插件 1.下载下面的插件 2.连接测试 3.生成代码的配置 4.生成代码 红色的是刚刚生成的。 我觉得不如官方的那个好用&#xff0c;唯一的好处就是勾选的选项能够看的懂得。

代码随想录第36天|435. 无重叠区间 (需要二刷),763.划分字母区间,56. 合并区间

435. 无重叠区间 &#xff08;需要二刷&#xff09; 本题其实和452.用最少数量的箭引爆气球 (opens new window)非常像&#xff0c;弓箭的数量就相当于是非交叉区间的数量&#xff0c;只要把弓箭那道题目代码里射爆气球的判断条件加个等号&#xff08;认为[0&#xff0c;1][1&a…

公司电脑文件自动加密——「天锐绿盾」

「天锐绿盾」是一种文件自动加密工具&#xff0c;可以与天锐绿盾应用服务器安全接入系统结合使用&#xff0c;实现只有安装了加密客户端的电脑才能访问应用服务器。此外&#xff0c;该工具还可以指定办公人员在对某些类型的文件进行新建、编辑时&#xff0c;自动备份到天锐绿盾…

没有使用sniffer dongle在windows抓包蓝牙方法分享

网上很多文章都是介绍买一个sniffer dongle来抓蓝牙数据,嫌麻烦又费钱,目前找到一个好方法,不需要sniffer就可以抓蓝牙数据过程,现分享如下: (1)在我资源附件找到相关安装包或者查看如下链接 https://learn.microsoft.com/zh-cn/windows-hardware/drivers/bluetooth/testing-bt…

2023开学礼《乡村振兴战略下传统村落文化旅游设计》许少辉八一新书海口经济学院图书馆

2023开学礼《乡村振兴战略下传统村落文化旅游设计》许少辉八一新书海口经济学院图书馆

通过Docker Compose安装MQTT

一、文件和目录说明 1、MQTT安装时的文件和目录 EMQX 安装完成后会创建一些目录用来存放运行文件和配置文件&#xff0c;存储数据以及记录日志。 不同安装方式得到的文件和目录位置有所不同&#xff0c;具体如下&#xff1a; 注意&#xff1a; 压缩包解压安装时&#xff0c;目…

第 3 章 栈和队列(顺序栈,算法 3.3,使用栈解答迷宫问题)

1. 背景说明 若迷宫 maze 中存在从入口 start 到出口 end 的通道&#xff0c;则求得一条存放在栈中(从栈底到栈顶)&#xff0c;并返回 TRUE&#xff1b;否则返回 FALSE&#xff0c;注意&#xff0c;该解并非最优解&#xff0c; 最优解需要求得最短路径且可能并非一条。 迷宫…

Auto DevOps介绍

1 Preface/Foreword 随着国际化&#xff0c;信息化&#xff0c;当今是一个VUCA时代。 VUCA&#xff1a;Volatile &#xff08;易变的&#xff09;&#xff0c;Uncertain &#xff08;不确定的&#xff09;&#xff0c;Complicated &#xff08;复杂的&#xff09;&#xff0c…

类ChatGPT大模型LLaMA及其微调模型

1.LLaMA LLaMA的模型架构:RMSNorm/SwiGLU/RoPE/Transfor mer/1-1.4T tokens 1.1对transformer子层的输入归一化 对每个transformer子层的输入使用RMSNorm进行归一化&#xff0c;计算如下&#xff1a; 1.2使用SwiGLU替换ReLU 【Relu激活函数】Relu(x) max(0,x) 。 【GLU激…