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;
}
相关推荐
不想写代码的星星2 小时前
虚函数表:C++ 多态背后的那个男人
c++
阿星AI工作室2 小时前
给openclaw龙虾造了间像素办公室!实时看它写代码、摸鱼、修bug、写日报,太可爱了吧!
前端·人工智能·设计模式
_哆啦A梦21 小时前
Vibe Coding 全栈专业名词清单|设计模式·基础篇(创建型+结构型核心名词)
前端·设计模式·vibecoding
端平入洛2 天前
delete又未完全delete
c++
端平入洛3 天前
auto有时不auto
c++
哇哈哈20214 天前
信号量和信号
linux·c++
多恩Stone4 天前
【C++入门扫盲1】C++ 与 Python:类型、编译器/解释器与 CPU 的关系
开发语言·c++·人工智能·python·算法·3d·aigc
蜡笔小马4 天前
21.Boost.Geometry disjoint、distance、envelope、equals、expand和for_each算法接口详解
c++·算法·boost
超级大福宝4 天前
N皇后问题:经典回溯算法的一些分析
数据结构·c++·算法·leetcode
weiabc4 天前
printf(“%lf“, ys) 和 cout << ys 输出的浮点数格式存在细微差异
数据结构·c++·算法