设计模式——观察者模式

定义

观察者模式(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中。

缺点

  • 可能会导致过多的通知开销:
    如果有大量的观察者对象,并且主题对象的状态频繁变化,那么通知所有观察者对象会产生较大的开销,可能会影响系统的性能。
  • 可能存在循环依赖问题:
    在复杂的系统中,如果观察者对象和主题对象之间存在循环依赖关系,可能会导致系统的逻辑混乱,难以维护。不过,通过合理的设计和分层可以尽量避免这种情况。
相关推荐
songroom3 分钟前
Rust: offset祼指针操作
开发语言·算法·rust
chenziang12 小时前
leetcode hot100 环形链表2
算法·leetcode·链表
Captain823Jack4 小时前
nlp新词发现——浅析 TF·IDF
人工智能·python·深度学习·神经网络·算法·自然语言处理
Captain823Jack4 小时前
w04_nlp大模型训练·中文分词
人工智能·python·深度学习·神经网络·算法·自然语言处理·中文分词
是小胡嘛5 小时前
数据结构之旅:红黑树如何驱动 Set 和 Map
数据结构·算法
m0_748255025 小时前
前端常用算法集合
前端·算法
呆呆的猫5 小时前
【LeetCode】227、基本计算器 II
算法·leetcode·职场和发展
Tisfy5 小时前
LeetCode 1705.吃苹果的最大数目:贪心(优先队列) - 清晰题解
算法·leetcode·优先队列·贪心·
余额不足121386 小时前
C语言基础十六:枚举、c语言中文件的读写操作
linux·c语言·算法
火星机器人life8 小时前
基于ceres优化的3d激光雷达开源算法
算法·3d