观察者模式是一种常见的软件设计模式,用于实现对象之间的一对多依赖关系。在观察者模式中,一个对象(称为主题或可观察对象)维护一个依赖列表,即一组观察者对象。当主题对象的状态发生变化时,它会自动通知所有观察者,使它们能够自动更新。
以下是观察者模式的基本组成部分:
① 主题(Subject):负责维护一组观察者对象,并提供注册、删除和通知观察者的方法。
② 观察者(Observer):定义一个更新方法,用于接收主题对象状态变化的通知。
③ 具体主题(ConcreteSubject):实现主题接口,维护其状态,并在状态发生变化时通知观察者。
④ 具体观察者(ConcreteObserver):实现观察者接口中的更新方法,并在收到通知时执行相应的操作。
观察者模式的优点包括:
- 松耦合性:主题对象和观察者对象之间的关系是松耦合的,它们之间并不直接依赖于彼此的具体实现。
- 可扩展性:可以轻松地添加新的观察者对象,而无需修改主题对象的代码。
- 封装变化:主题对象和观察者对象之间的通信逻辑被封装在了观察者模式中,使得系统更易于维护和修改。
观察者模式在软件开发中被广泛应用,特别是在需要实现对象间的一对多关系,并且希望避免对象之间紧密耦合的情况下。

1、场景设计

实现场景:定义一个主题,主题可以添加或删除观察者,主题可以通知数据变化,被添加(且未被删除)的观察者可以观察到主题的通知的数据变化。

2、C++实现

`ConcreteSubject` 类代表具体的主题,`ConcreteObserver` 类代表具体的观察者。主题维护了一个观察者列表,可以通过 `attach` 方法添加观察者,通过 `detach` 方法移除观察者,通过 `notify` 方法通知观察者。当主题状态发生变化时,它会调用 `notify` 方法通知所有注册的观察者。

#include <iostream>
#include <vector>
#include <algorithm>

// 观察者基类
class Observer {
    public:
        virtual void update(int data) = 0;
};

// 具体观察者类
class ConcreteObserver : public Observer {
    public:
        void update(int data) override {
            std::cout << "ConcreteObserver received update with data: " << data << std::endl;
        }
};

// 主题基类
class Subject {
    public:
        virtual void attach(Observer* observer) = 0;
        virtual void detach(Observer* observer) = 0;
        virtual void notify(int data) = 0;
};

// 具体主题类
class ConcreteSubject : public Subject {
    private:
        std::vector<Observer*> observers;

    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(int data) override {
            for (auto observer : observers) {
                observer->update(data);
            }
        }
};

int main() {
    ConcreteSubject subject;

    ConcreteObserver observer1;
    ConcreteObserver observer2;

    subject.attach(&observer1);
    subject.attach(&observer2);

    subject.notify(10);

    subject.detach(&observer1);

    subject.notify(20);

    return 0;
}

3、Java实现

`ConcreteSubject` 类代表具体的主题,`ConcreteObserver` 类代表具体的观察者。主题维护了一个观察者列表,可以通过 `attach` 方法添加观察者,通过 `detach` 方法移除观察者,通过 `notify` 方法通知观察者。当主题状态发生变化时,它会调用 `notify` 方法通知所有注册的观察者。 

package behavioralpattern.observer;
import java.util.ArrayList;
import java.util.List;
public class ObserverDemo {
    // 观察者接口
    interface Observer {
        void update(int data);
    }

    // 具体观察者类
    static class ConcreteObserver implements Observer {
        @Override
        public void update(int data) {
            System.out.println("ConcreteObserver received update with data: " + data);
        }
    }

    // 主题接口
    interface Subject {
        void attach(Observer observer);
        void detach(Observer observer);
        void notify(int data);
    }

    // 具体主题类
    static class ConcreteSubject implements Subject {
        private List<Observer> observers = new ArrayList<>();

        @Override
        public void attach(Observer observer) {
            observers.add(observer);
        }

        @Override
        public void detach(Observer observer) {
            observers.remove(observer);
        }

        @Override
        public void notify(int data) {
            for (Observer observer : observers) {
                observer.update(data);
            }
        }
    }

    public static void main(String[] args) {
        ConcreteSubject subject = new ConcreteSubject();

        ConcreteObserver observer1 = new ConcreteObserver();
        ConcreteObserver observer2 = new ConcreteObserver();

        subject.attach(observer1);
        subject.attach(observer2);

        subject.notify(10);

        subject.detach(observer1);

        subject.notify(20);
    }
}

 

04-17 15:36