设计模式——观察者模式

定义

观察者模式(Observer Pattern)是一种行为型设计模式。它定义了一种一对多的依赖关系,让多个观察者对象(Observer)同时监听一个主题对象(Subject)。当主题对象的状态发生变化时,它会通知所有依赖它的观察者对象,观察者对象会根据主题对象状态的变化做出相应的反应。

UML 类图及主要角色

  • 抽象主题(Subject):
    它是被观察的对象,一般是一个抽象类或接口。它维护了一个观察者对象的列表,并且提供了添加、删除观察者对象的方法,同时还包括一个通知所有观察者对象的抽象方法。
  • 具体主题(Concrete Subject):
    实现了抽象主题中的方法。它管理主题对象的状态,当状态发生变化时,它会调用通知方法来告知所有的观察者对象。
  • 抽象观察者(Observer):
    定义了一个抽象的更新方法,当接收到主题对象的通知时,观察者对象通过这个方法来更新自己的状态。
  • 具体观察者(Concrete Observer):
    实现了抽象观察者的更新方法。每个具体观察者对象根据自己的需求来处理主题对象状态变化的信息。

简单示例:新闻发布系统

  • 抽象主题类
cpp 复制代码
class NewsSubject {
public:
  virtual ~NewsSubject() {}
  virtual void attach(Observer* observer) = 0;
  virtual void detach(Observer* observer) = 0;
  virtual void notify() = 0;
};
  • 具体主题类
cpp 复制代码
class NewsAgency : public NewsSubject {
private:
  std::vector<Observer*> observers;
  std::string news;
public:
  void attach(Observer* observer) override {
    observers.push_back(observer);
  }
  void detach(Observer* observer) override {
    auto it = std::find(observers.begin(), observers.end(), observer);
    if (it!= observers.end()) {
      observers.erase(it);
    }
  }
  void notify() override {
    for (Observer* observer : observers) {
      observer->update(news);
    }
  }
  void setNews(const std::string& n) {
    news = n;
    notify();
  }
};
  • 抽象观察者类
cpp 复制代码
class Observer {
public:
  virtual ~Observer() {}
  virtual void update(const std::string& news) = 0;
};

具体观察者类

cpp 复制代码
class NewsReader : public Observer {
private:
  std::string name;
public:
  NewsReader(const std::string& n) : name(n) {}
  void update(const std::string& news) override {
    std::cout << name << " received news: " << news << std::endl;
  }
};

使用示例

cpp 复制代码
int main() {
  NewsAgency agency;
  NewsReader reader1("Reader1");
  NewsReader reader2("Reader2");
  agency.attach(&reader1);
  agency.attach(&reader2);
  agency.setNews("Breaking news: A new discovery!");
  agency.detach(&reader1);
  agency.setNews("Another news: Technology update.");
  return 0;
}

在这个示例中,NewsSubject是抽象主题,NewsAgency是具体主题,它管理新闻内容并且通知观察者。Observer是抽象观察者,NewsReader是具体观察者,它接收新闻并进行显示。

优点

  • 松耦合:
    主题和观察者之间是松耦合的关系。主题对象不需要知道观察者对象的具体细节,只需要在状态变化时通知它们即可。同样,观察者对象也只需要实现更新方法来处理主题对象状态变化的信息,不需要了解主题对象内部是如何维护状态的。
  • 可扩展性好:
    可以很容易地添加新的观察者对象,只要实现抽象观察者的更新方法即可。在上述新闻发布系统的例子中,如果要添加新的新闻读者,只需要创建一个新的NewsReader对象并将其添加到NewsAgency中。

缺点

  • 可能会导致过多的通知开销:
    如果有大量的观察者对象,并且主题对象的状态频繁变化,那么通知所有观察者对象会产生较大的开销,可能会影响系统的性能。
  • 可能存在循环依赖问题:
    在复杂的系统中,如果观察者对象和主题对象之间存在循环依赖关系,可能会导致系统的逻辑混乱,难以维护。不过,通过合理的设计和分层可以尽量避免这种情况。
相关推荐
历程里程碑2 分钟前
C++ 7vector:动态数组的终极指南
java·c语言·开发语言·数据结构·c++·算法
mit6.8244 分钟前
get+二分|数位dp
算法
sin_hielo8 分钟前
leetcode 2147
数据结构·算法·leetcode
萌>__<新21 分钟前
力扣打卡每日一题——缺失的第一个正数
数据结构·算法·leetcode
DuHz25 分钟前
车对车对向交汇场景的毫米波路径损耗建模论文精读
论文阅读·算法·汽车·信息与通信·信号处理
lxh011337 分钟前
二叉树中的最大路径和
前端·算法·js
萌>__<新39 分钟前
力扣打卡每日一题————零钱兑换
算法·leetcode·职场和发展
重生之后端学习39 分钟前
238. 除自身以外数组的乘积
java·数据结构·算法·leetcode·职场和发展·哈希算法
yaoxin5211231 小时前
269. Java Stream API - Map-Filter-Reduce算法模型
java·python·算法
Learner__Q1 小时前
每天五分钟:动态规划-LeetCode高频题_day2
算法·leetcode·动态规划