设计模式——观察者模式

定义

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

缺点

  • 可能会导致过多的通知开销:
    如果有大量的观察者对象,并且主题对象的状态频繁变化,那么通知所有观察者对象会产生较大的开销,可能会影响系统的性能。
  • 可能存在循环依赖问题:
    在复杂的系统中,如果观察者对象和主题对象之间存在循环依赖关系,可能会导致系统的逻辑混乱,难以维护。不过,通过合理的设计和分层可以尽量避免这种情况。
相关推荐
向宇it1 分钟前
【unity小技巧】Unity 四叉树算法实现空间分割、物体存储并进行查询和碰撞检测
开发语言·算法·游戏·unity·游戏引擎
无限大.1 分钟前
冒泡排序(结合动画进行可视化分析)
算法·排序算法
repinkply12 分钟前
装饰模式(Decorator)模式:装饰模式/包装模式,结构模式
设计模式·c++设计模式·装饰器模式·装饰模式·包装模式
走向自由21 分钟前
Leetcode 最长回文子串
数据结构·算法·leetcode·回文·最长回文
nuo53420235 分钟前
The 2024 ICPC Kunming Invitational Contest
c语言·数据结构·c++·算法
luckilyil36 分钟前
Leetcode 每日一题 11. 盛最多水的容器
算法·leetcode
霁月风1 小时前
设计模式——装饰器模式
c++·设计模式·装饰器模式
夏旭泽1 小时前
设计模式-原型模式
设计模式·原型模式
A.A呐1 小时前
LeetCode 1658.将x减到0的最小操作数
算法·leetcode
hn小菜鸡1 小时前
LeetCode 144.二叉树的前序遍历
算法·leetcode·职场和发展