设计模式学习笔记-观察者模式

观察者模式

个人博客:nezha.github.io
我的公众号:nezha_blog

本文的源代码放在我的GitHub上:nezha/DesignPatterns

观察者模式是对象的行为模式,又叫发布-订阅(Publish/Subscribe)模式、模型-视图(Model/View)模式、源-监听器(Source/Listener)模式或从属者(Dependents)模式。

观察者模式定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象。这个主题对象在状态上发生变化时,会通知所有观察者对象,使它们能够自动更新自己。

Java Messages Service(JMS)消息服务使用观察者模式与命令模式来实现不同的程序之间的数据的发布和订阅。

观察者模式的结构

Command + option + shift

《设计模式学习笔记-观察者模式》
image

观察者模式所涉及的角色有:

  • 抽象主题(Subject)角色:抽象主题角色把所有对观察者对象的引用保存在一个聚集(比如ArrayList对象)里,每个主题都可以有任何数量的观察者。抽象主题提供一个接口,可以增加和删除观察者对象,抽象主题角色又叫做抽象被观察者(Observable)角色。

  • 具体主题(ConcreteSubject)角色:将有关状态存入具体观察者对象;在具体主题的内部状态改变时,给所有登记过的观察者发出通知。具体主题角色又叫做具体被观察者(Concrete Observable)角色。

  • 抽象观察者(Observer)角色:为所有的具体观察者定义一个接口,在得到主题的通知时更新自己,这个接口叫做更新接口。

  • 具体观察者(ConcreteObserver)角色:存储与主题的状态自恰的状态。具体观察者角色实现抽象观察者角色所要求的更新接口,以便使本身的状态与主题的状态 像协调。如果需要,具体观察者角色可以保持一个指向具体主题对象的引用。

观察者模式例子

1.抽象主题(Subject)

public interface Subject {
    //methods to register and unregister observers
    public void register(Observer obj);
    public void unregister(Observer obj);
    //method to notify observers of change
    public void notifyObservers();
}

2.具体主题(ConcreteSubject)

public class ConcreteSubject implements Subject {

    private List<Observer> observers;
    private String message;
    private boolean changed;
    private final Object MUTEX= new Object();

    public ConcreteSubject(){
        this.observers=new ArrayList<Observer>();
    }
    @Override
    public void register(Observer obj) {
        if(obj == null) throw new NullPointerException("Null Observer");
        if(!observers.contains(obj)) observers.add(obj);
    }

    @Override
    public void unregister(Observer obj) {
        observers.remove(obj);
    }

    @Override
    public void notifyObservers() {
        List<Observer> observersLocal = null;
        //synchronization is used to make sure any observer registered after message is received is not notified
        synchronized (MUTEX) {
            if (!changed)
                return;
            observersLocal = new ArrayList<>(this.observers);
            this.changed=false;
        }
        for (Observer obj : observersLocal) {
            obj.update("" + new Date() + ">>>"+this.message);
        }

    }
    //method to post message to the topic
    public void postMessage(String msg){
        System.out.println("Message Posted to Subject:"+msg);
        this.message=msg;
        this.changed=true;
        notifyObservers();
    }
}

3.抽象观察者(Observer)

public interface Observer {
    //method to update the observer, used by subject
    public void update(String content);
}

4.具体观察者(ConcreteObserver)

public class ConcreteObserver implements Observer {

    private String name;
    private Subject topic;
    private String content;

    public ConcreteObserver(Subject topic, String nm){
        this.name=nm;
        this.topic=topic;
        topic.register(this);
    }
    @Override
    public void update(String content) {
        this.content = content;
        System.out.println(name + this.getClass().getName() +"--- update the info:"+this.content);
    }
}

5.测试实验

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

        //create observers
        Observer obj1 = new ConcreteObserver(subject,"Obj1");
        Observer obj2 = new ConcreteObserver(subject,"Obj2");
        Observer obj3 = new ConcreteObserver(subject,"Obj3");
        //now send message to subject
        subject.postMessage("New Message");
        //这里是解绑某一个对象,主动权在subject手中
        subject.unregister(obj2);

        subject.postMessage("将二号对象移除");
    }
}

参考文献

ifeve.com/observer-de…

    原文作者:算法小白
    原文地址: https://juejin.im/entry/5a2921e8f265da432d27ffe2
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞