定义
观察者模式(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中。
缺点
- 可能会导致过多的通知开销:
如果有大量的观察者对象,并且主题对象的状态频繁变化,那么通知所有观察者对象会产生较大的开销,可能会影响系统的性能。 - 可能存在循环依赖问题:
在复杂的系统中,如果观察者对象和主题对象之间存在循环依赖关系,可能会导致系统的逻辑混乱,难以维护。不过,通过合理的设计和分层可以尽量避免这种情况。