观察者模式(Observer) 又被称"发布-订阅模式"

一:场景说明

记得上初中时,网络还没有现在这么普及,校门口有个小卖部,我常常到哪里订阅报纸,来关心下“天下大事”(当然不止我一个人订阅啦)

如图订阅报纸流程:

观察者模式(Observer) 又被称"发布-订阅模式"_第1张图片

订阅报纸的时候常常遇到几个问题?什么时候出版?出版内容是什么?延迟出版怎么办?谁通知我报纸到了?要我去取报纸?还是会有人送过来?等等

把这样的场景带到我们编程世界里来,我们的解决方案之一就是使用观察者设计模式。

二:观察者模式(Observer),定义对象间的一种一对多的依赖关系。当一个对象的状态或内容发生改变时,所有依赖它的对象都得到通知并被自动更新.

观察者模式的结构说明:

观察者模式(Observer) 又被称"发布-订阅模式"_第2张图片

观察者模式(Observer) 又被称"发布-订阅模式"_第3张图片


三:代码示例:

其实JDK已经实现的Subject和Observer接口的,我们直接可以使用,下面我会介绍的,这里还是先自己实现下,体会下吧


/**
 * 观察者接口
 * 提供通知更新方法 (可以在这个方法中回调目标对象)
 * Created by Administrator on 2016/4/22.
 */
public interface Observer {
    void update(Subject subject);
}



package com.dy.observer;

import java.util.ArrayList;
import java.util.List;


/**
 * 目标对象 ,作为被观察者
 * 它知道观察者,并提供注册和删除观察者
 * Created by Administrator on 2016/4/22.
 */
public class Subject {

    private List<Observer> observerList = new ArrayList<>();

    /**
     * 注册观察者 ,报纸订阅者
     * @param observer
     */
    public void attach(Observer observer){
        observerList.add(observer);
    }

    /**
     * 删除观察者
     * @param observer
     */
    public void detach(Observer observer){
        observerList.remove(observer);
    }

    /**
     * 通知所有观察者
     * 每当报纸被印出来,及时主动通知所有订阅者,好让他们知道
     */
    protected void notifyObserver(){
        for (Observer observer:observerList){
            observer.update(this);
        }
    }
}



package com.dy.observer;

/**
 * 目标对象的继承者
 * 负责具体观察者状态
 * 当自己状态变化,通知所有观察者
 * Created by Administrator on 2016/4/22.
 */
public class NewsPaper extends Subject {
    private String subjectStatus;
    /**
     * 报纸内容
     */
    private String content;

    public String getSubjectStatus() {
        return subjectStatus;
    }


    public void setSubjectStatus(String subjectStatus) {
        this.subjectStatus = subjectStatus;
        //状态改变,通知所有观察者
        this.notifyObserver();
    }

    public String getContent() {
        return content;
    }

    public void setContent(String content) {
        this.content = content;
        //内容有了,说明报纸被印刷出来了,通知所有观察者
        this.notifyObserver();
    }
}



package com.dy.observer;

/**
 * 读者,报纸订阅者
 *
 * 观察者实现类
 * 实现更新方法,使自身状态和目标状态保持一致
 * Created by Administrator on 2016/4/22.
 */
public class Reader implements Observer {


    private String observerStatus;
    private String name;

    public Reader(String name) {
        this.name = name;
    }

    @Override
    public void update(Subject subject) {
        //实现更新方法,使自身状态和目标状态保持一致
        //采用拉的方式
        observerStatus = ((NewsPaper) subject).getSubjectStatus();
        System.out.println(name+"收到了报纸,阅读它,内容是:"+((NewsPaper) subject).getContent());
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}



package com.dy.observer;

/**
 * Created by Administrator on 2016/4/22.
 */
public class Client {
    public static void main(String[] args) {

        //报纸被观察者
        NewsPaper newsPager = new NewsPaper();

        //报纸订阅者,也就是观察者
        Reader reader = new Reader("张三");
        Reader reader1 = new Reader("李四");
        Reader reader2 = new Reader("王二");

        //注册订阅者
        newsPager.attach(reader);
        newsPager.attach(reader1);
        newsPager.attach(reader2);

        //印刷出报纸内容
        newsPager.setContent("本期内容介绍观察者模式");
    }
}



客户端测试结果:
张三收到了报纸,阅读它,内容是:本期内容介绍观察者模式
李四收到了报纸,阅读它,内容是:本期内容介绍观察者模式
王二收到了报纸,阅读它,内容是:本期内容介绍观察者模式

Process finished with exit code 0


四:使用jdk实现观察者模式

注意修改对应代码


package com.dy.observer.java;

import java.util.Observable;

/**
 * Created by Administrator on 2016/4/22.
 */
public class NewsPaper extends Observable {

    /**
     * 报纸内容
     */
    private String content;

    public String getContent() {
        return content;
    }

    public void setContent(String content) {
        this.content = content;
        //内容有了,说明报纸被印刷出来了,通知所有观察者

        //注意JDK的observer模式的时候,下面这句代码不可少
        this.setChanged();
        //主动通知,这里用推的方式
        this.notifyObservers(this.content);
        //如果用的拉的方式,如下代码示例
        //this.notifyObservers();
    }

}



package com.dy.observer.java;


import java.util.Observable;
import java.util.Observer;

/**
 * 读者,报纸订阅者
 *
 * 观察者实现类
 * 实现更新方法,使自身状态和目标状态保持一致
 * Created by Administrator on 2016/4/22.
 */
public class Reader implements Observer {

    private String name;

    public Reader(String name) {
        this.name = name;
    }
    @Override
    public void update(Observable o, Object obj) {
        System.out.println(name+"收到了报纸,目标推过来的内容是:"+obj);
        //拉的方式你自己想想就知道了((NewsPaper) obj).getContent())
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}



public class Client {
    public static void main(String[] args) {

        //报纸被观察者
        NewsPaper jdk_subject = new NewsPaper();

        //报纸订阅者,也就是观察者
        Reader reader = new Reader("张三");
        Reader reader1 = new Reader("李四");
        Reader reader2 = new Reader("王二");

        //注册订阅者
        jdk_subject.addObserver(reader);
        jdk_subject.addObserver(reader1);
        jdk_subject.addObserver(reader2);

        //印刷出报纸内容
        jdk_subject.setContent("本期内容介绍java JDK实现观察者模式");
    }
}



王二收到了报纸,目标推过来的内容是:本期内容介绍java JDK实现观察者模式
李四收到了报纸,目标推过来的内容是:本期内容介绍java JDK实现观察者模式
张三收到了报纸,目标推过来的内容是:本期内容介绍java JDK实现观察者模式

Process finished with exit code 0




你可能感兴趣的:(观察者模式(Observer))