首页 > 编程语言 >《C++20设计模式》观察者模式

《C++20设计模式》观察者模式

时间:2024-07-10 10:28:31浏览次数:26  
标签:20 name observer lock C++ shared 设计模式 Event const

一、前言

观察者模式感觉真的很难,我这里就实现书上差不多的例子,供大家学习吧!

相关代码可以在这里,如有帮助给个star!AidenYuanDev/design_patterns_in_modern_Cpp_20

二、实现

  1. 这里说一下观察者模式是干什么的。

在实际中很有用,比如以下的例子。

  • 当股票价格变化时,自动通知所有关注该股票的观察者
  • 当有新闻发布时,自动向所有订阅用户推送通知
  1. 大概是怎么实现的

如果一个类中的属性发生变化了,他将通知观察者,然后让观察者来通知。

1、UML类图

观察者模式

2、实现

#include <algorithm>
#include <any>
#include <iostream>
#include <memory>
#include <mutex>
#include <shared_mutex>
#include <unordered_map>
#include <vector>
using namespace std;

enum class Event { NameChanged, AgeChanged, ScoreChanged, StatusChanged };

class Observer_Base {
protected:
  string observer_name;
  shared_mutex mutex;

public:
    Observer_Base(const string &observer_name) : observer_name(observer_name) {}
    virtual void update(Event event, const any &value) = 0;
};

class Person : public enable_shared_from_this<Person> {
private:
    string name;
    int age;
    double score;
    bool active;

    shared_mutex mutex;
    unordered_map<Event, vector<weak_ptr<Observer_Base>>> observers;

private:
    void notify(Event event, const any &value) {
        shared_lock lock(mutex);
        auto it = observers.find(event);
        if (it != observers.end()) {
            auto observers_copy = it->second;
            lock.unlock();
            for (auto &weak_obs : observers_copy) {
                if (auto obs = weak_obs.lock()) obs->update(event, value);
            }
        }
    }

public:
    Person(const string &name, const int &age, const double &score, const bool active) : name(name), age(age), score(score), active(active) {}

    auto subscribe(Event event, shared_ptr<Observer_Base> observer) {
        unique_lock lock(mutex);
        observers[event].push_back(observer);
        return shared_from_this();
    }

    auto unsubscribe(Event event, shared_ptr<Observer_Base> observer) {
        unique_lock lock(mutex);
        auto &obs = observers[event];
        obs.erase(remove_if(obs.begin(), obs.end(), [&](const weak_ptr<Observer_Base> &wp) { return wp.lock() == observer; }), obs.end());
        return shared_from_this();
    }

    auto set_name(const string &name_new) {
        unique_lock lock(mutex);
        if (name != name_new) {
            name = name_new;
            lock.unlock();
            notify(Event::NameChanged, name);
        }
        return shared_from_this();
    }

    auto set_age(const int &age_new) {
        unique_lock lock(mutex);
        if (age != age_new) {
            age = age_new;
            lock.unlock();
            notify(Event::AgeChanged, age);
        }
        return shared_from_this();
    }

    auto set_score(const double &score_new) {
        unique_lock lock(mutex);
        if (score != score_new) {
            score = score_new;
            lock.unlock();
            notify(Event::ScoreChanged, score);
        }
        return shared_from_this();
    }

    auto set_Active(const bool &status_new) {
        unique_lock lock(mutex);
        if (active != status_new) {
            active = status_new;
            lock.unlock();
            notify(Event::StatusChanged, active);
        }
        return shared_from_this();
    }
};

// 具体观察者类
class Person_Oberver : public Observer_Base {
public:
    Person_Oberver(const string &name) : Observer_Base(name) {}

    void update(Event event, const any &value) override {
        unique_lock lock(mutex);
        switch (event) {
            case Event::NameChanged:
                cout << observer_name << " observed name change to: " << any_cast<string>(value) << endl;
                break;
            case Event::AgeChanged:
                std::cout << observer_name << " observed age change to: " << any_cast<int>(value) << endl;
                break;
            case Event::ScoreChanged:
                std::cout << observer_name << " observed score change to: " << any_cast<double>(value) << endl;
                break;
            case Event::StatusChanged:
                std::cout << observer_name << " observed status change to: " << (any_cast<bool>(value) ? "active" : "inactive") << endl;
                break;
        }
    }
};

int main() {
    auto person = make_shared<Person>("Starry Decade", 18, 98, false);
    auto observer_1 = make_shared<Person_Oberver>("observer 1");
    auto observer_2 = make_shared<Person_Oberver>("observer 2");

    person->subscribe(Event::NameChanged, observer_1)
          ->subscribe(Event::AgeChanged, observer_1)
          ->subscribe(Event::ScoreChanged, observer_2)
          ->subscribe(Event::StatusChanged, observer_2);
    
    person->set_name("Aiden")
          ->set_age(19)
          ->set_score(100.)
          ->set_Active(true);

    return 0;
}

标签:20,name,observer,lock,C++,shared,设计模式,Event,const
From: https://blog.csdn.net/Ten_years_star/article/details/140273451

相关文章

  • 2024 「全球软件研发技术大会】-刘兴东分享京东的AIGC革新之旅
    大模型和开源的发展将带来全球软件研发技术的新变革,AI使代码自动化应用达到新水平,开源工具的云化和应用的AI化将促中国软件迎来新一轮的爆发。开发者正在迎接新一轮的技术浪潮变革。由CSDN和高端IT咨询和教育平台Boolan联合主办的2024年度「全球软件研发技术大会」于7月4日-5日在......
  • 【免费】最新最详细的华为OD2024机试题,ABCD卷,有答案,很全,很新
    最新最详细的华为OD2024机试题,ABCD卷,有答案,很全,很新!+目前461题,亲测资料非常完整好用!!最近考试换为CD卷,CD卷题库是一样的,D卷为双机位监控,某些外包公司应聘的为D卷。其中资料中JAVA,python,+JavaScript,C语言都有包含的,绝对会对你有所帮助!!获取方式:https://pan.quark.cn/s/9b122f......
  • 2024程序员行业风口和面试宝典
    国际研究机构Gartner会在每年10月份左右发布下一年度的战略发展趋势预测,并在次年3月左右发布和网络安全相关的趋势预测。绿盟科技通过将近3年的趋势预测进行分组对比分析后发现,除了众人皆知的AI技术应用外,数据模块化、身份优先安全、行业云平台也可能会成为未来网络安全领......
  • 电影《云边有个小卖部》迅雷BT下载[MP4/2.12GB/5.35GB]高清画质[HD720p/1080p]百度云
    电影《云边有个小卖部》是由中国导演张嘉佳执导,于2024年上映的一部浪漫爱情片。影片以小卖部为背景,讲述了一个平凡的小镇故事,通过描绘人物的情感故事和生活琐事,展现出生活的真实与温情。 影片的故事发生在一个偏远的小山村,村民们日复一日地过着平凡而宁静的生活。主......
  • 计算机毕业设计项目:18655 课程题库管理系统(开题答辩+程序定制+全套文案 )上万套实战教
    摘 要随着科学技术的飞速发展,各行各业都在努力与现代先进技术接轨,通过科技手段提高自身的优势;对于课程题库管理系统当然也不能排除在外,随着网络技术的不断成熟,带动了课程题库管理系统,它彻底改变了过去传统的管理方式,不仅使服务管理难度变低了,还提升了管理的灵活性。这种个......
  • 计算机毕业设计项目: node.js 网上购物商城的设计与实现99525(开题答辩+程序定制+全套文
    摘 要随着社会的发展,计算机的优势和普及使得网上购物商城的开发成为必需。网上购物商城主要是借助计算机,通过对首页、站点管理(轮播图、公告栏)用户管理(管理员、注册用户)内容管理(商城资讯、资讯分类)商城管理(商城中心、分类列表、订单列表)等信息进行管理。减少管理员的工作......
  • 计算机毕业设计项目:校园新闻数据化系统的设计与实现 99293(开题答辩+程序定制+全套文案
    目 录摘要1绪论1.1研究目的和意义1.2选题背景和意义1.3系统开发技术的特色1.4springboot框架介绍1.5论文结构与章节安排22校园新闻数据化系统系统分析2.1可行性分析2.2系统流程分析2.2.1数据增加流程2.3.2数据修改流程2.3.3数据删除流程2.3......
  • 20240710概率期望
    概率基础知识不写了,反正应该知道的都知道但是有几个跟容斥有关的不知道,我要记录下1.互斥事件可加性:对于n个互斥的事件\(P(A_1\cup...\cupA_n)=\sum_{i=1}^{n}A_i\)2.独立事件可乘性:对于n个对立的事件\(P(A_1\cap...\capA_n)=\prod_{i=1}^{n}A_i\)3.n重伯努利实验:一次实验......
  • 第三章 设计模式(2023版本IDEA)
    学习目标3.1设计模式概述3.2软件可复用问题和面向对象设计原则一、软件可复用问题二、面向对象设计原则1.单一责任原则(SingleResponsibilityPrinciple,SRP)2.开放-封闭原则(Open-ClosedPrinciple,OCP)3.里氏替换原则(LiskovSubstitutionPrinciple,LSP)4.依赖倒......
  • C++类和对象(上篇)
    文章目录前言一、面向过程和面向对象初步认识二、类的引入三、类的定义六、类的实例化七、类的对象大小的计算八、类成员函数的this指针总结前言类和对象是面向对象编程的两个核心概念。类是一种抽象的数据类型,是描述对象共同特......