C++ 设计模式:观察者模式(Observer Pattern)

链接:C++ 设计模式
链接:C++ 设计模式 - 模板方法
链接:C++ 设计模式 - 策略模式

观察者模式(Observer Pattern)是一种行为设计模式,它定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象。当主题对象的状态发生变化时,它会通知所有观察者对象,使它们能够自动更新。

1.问题分析

在开发中,经常会遇到需要在一个对象状态变化时通知其他对象的情况。如果将通知逻辑直接嵌入到主题对象中,会导致代码耦合度高,难以维护和扩展。观察者模式通过将观察者对象和主题对象解耦,使得它们可以独立变化。

2.实现步骤

  1. 定义观察者接口:定义一个接口,包含更新方法。
  2. 定义主题接口:定义一个接口,包含注册、注销和通知观察者的方法。
  3. 实现具体主题类:具体主题类实现主题接口,并维护一个观察者列表。
  4. 实现具体观察者类:具体观察者类实现观察者接口,并在更新方法中实现具体的响应逻辑。

3.代码示例

以机器人电池电量作为示例:

3.1.观察者接口

cpp 复制代码
// 定义更新方法
class Observer {
 public:
  virtual ~Observer() = default;
  virtual void update(int batteryLevel) = 0;
};

3.2. 主题接口

cpp 复制代码
// 定义添加、删除和通知观察者的方法
class Subject {
 public:
  virtual ~Subject() = default;
  virtual void addObserver(std::shared_ptr<Observer> observer) = 0;
  virtual void removeObserver(std::shared_ptr<Observer> observer) = 0;
  virtual void notifyObservers() = 0;
};

3.3.具体主题类

cpp 复制代码
// 实现主题接口
class Robot : public Subject {
 public:
  void addObserver(std::shared_ptr<Observer> observer) override { observers.push_back(observer); }

  void removeObserver(std::shared_ptr<Observer> observer) override {
    observers.erase(std::remove(observers.begin(), observers.end(), observer), observers.end());
  }

  void notifyObservers() override {
    for (const auto& observer : observers) {
      observer->update(batteryLevel);
    }
  }

  void setBatteryLevel(int level) {
    batteryLevel = level;
    notifyObservers();
  }

 private:
  std::vector<std::shared_ptr<Observer>> observers;
  int batteryLevel = 100;
};

3.4.具体观察者类

cpp 复制代码
// 具体观察者类1,显示电池电量
class BatteryDisplay : public Observer {
 public:
  void update(int batteryLevel) override { std::cout << "BatteryDisplay: Battery level is " << batteryLevel << "%" << std::endl; }
};
cpp 复制代码
// 具体观察者类2,当电池电量低于20%时发出警告
class BatteryWarning : public Observer {
 public:
  void update(int batteryLevel) override {
    if (batteryLevel < 20) {
      std::cout << "BatteryWarning: Warning - Battery level low (" << batteryLevel << "%)" << std::endl;
    }
  }
};

3.5.调用算法

cpp 复制代码
int main() {
  auto robot = std::make_shared<Robot>();
  auto batteryDisplay = std::make_shared<BatteryDisplay>();
  auto batteryWarning = std::make_shared<BatteryWarning>();

  robot->addObserver(batteryDisplay);
  robot->addObserver(batteryWarning);

  std::cout << "Setting battery level to 50%" << std::endl;
  robot->setBatteryLevel(50);

  std::cout << "\nSetting battery level to 15%" << std::endl;
  robot->setBatteryLevel(15);

  return 0;
}
相关推荐
郝学胜-神的一滴1 小时前
Linux命名管道:创建与原理详解
linux·运维·服务器·开发语言·c++·程序人生·个人开发
晚风(●•σ )1 小时前
C++语言程序设计——11 C语言风格输入/输出函数
c语言·开发语言·c++
恒者走天下3 小时前
秋招落定,拿到满意的offer,怎么提高自己实际的开发能力,更好的融入团队
c++
兵bing3 小时前
设计模式-访问者模式
设计模式·访问者模式
python零基础入门小白3 小时前
【万字长文】大模型应用开发:意图路由与查询重写设计模式(从入门到精通)
java·开发语言·设计模式·语言模型·架构·大模型应用开发·大模型学习
天若有情6733 小时前
【c++】手撸C++ Promise:从零实现通用异步回调组件,支持链式调用+异常安全
开发语言·前端·javascript·c++·promise
学困昇3 小时前
C++中的异常
android·java·c++
MC丶科4 小时前
Java设计模式漫画英雄宇宙-工厂模式 —Factory博士的“超级英雄制造机”!
java·设计模式·漫画
合作小小程序员小小店4 小时前
桌面安全开发,桌面二进制%恶意行为拦截查杀%系统安全开发3.0,基于c/c++语言,mfc,win32,ring3,dll,hook,inject,无数据库
c语言·开发语言·c++·安全·系统安全
Codeking__4 小时前
C++ 11 atomic 原子性操作
开发语言·c++