观察者模式--彻底理解运用Java设计模式之观察者模式

    定义与意义:

      观察者模式是对象的行为模式,它定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象。当主题对象在状态上发生变化时,会通知所有观察者对象,使它们能够自动更新自己。 我们又把这种模式成为发布--订阅模式(Publish/Subscribe)或者模型-视图模式(Model/View)模式。该模式的最重要作用是解耦,将被观察者和观察者进行解耦,使得它们的依赖性更小,甚至做到无依赖。

     实例实现

       我们假定一个场景,在一个图书管理系统中,现在有这么一个需求,部分读者希望在图书馆添加新书的时候能够收到通知。于是这些读者就订阅该功能。而当图书馆添加新书的时候,系统会自动将新书的信息发送给订阅的读者。 
  上面这个场景就是典型的观察者模式。对于读者我们可以称为观察者,而图书管理系统则可以作为被观察者。在图书管理系统(被观察者)中去添加注册需要接收通知的读者(观察者)。这时候在系统中添加新书后,系统便会通知在系统注册该功能的读者。下面看一下代码实现。 

        首先我们创建一个观察者接口。

package demo.springboot.web;

/**
 * @author 
 * @date 2018/12/25 16:08
 */
//观察者接口
public interface Observer {
    public void update(Object object);
}

         紧接着我们在创建一个被观察者,对于被观察者,里面的功能实现必然有添加一个观察者,删除观察者,通知更新观察者这三个公共方法,于是我们可以写一个抽象类。

package demo.springboot.web;

import java.util.Vector;

/**
 * @author 
 * @date 2018/12/25 16:09
 */
//被观察者的抽象类
public abstract class AbsObservable {
    //定义一个观察者数组
    private Vector<Observer> obVector = new Vector<>();

    //添加观察者
    public void addObserver(Observer ob){
        this.obVector.add(ob);
    }
    //删除观察者
    public void  deleObserver(Observer ob){
        this.obVector.remove(ob);
    }
    //通知所有观察者
    public void notifyAllObserver(Book book){
        for(Observer obs : obVector){
            obs.update(book);
        }

    }
}

下面我们就来实现上述场景。既然是图书管理系统,我们首先需要创建一个Book类(上面抽象类中的Book引用的即为即将创建的类)。

package demo.springboot.web;

/**
 * 
 * @date 2018/12/25 16:19
 */
public class Book {
    //书名
    public String bookName;
    //作者
    public String author;

    public String getBookName() {
        return bookName;
    }

    public void setBookName(String bookName) {
        this.bookName = bookName;
    }

    public String getAuthor() {
        return author;
    }

    public void setAuthor(String author) {
        this.author = author;
    }

    public Book(String bookName, String author) {
        this.bookName = bookName;
        this.author = author;
    }

    @Override
    public String toString() {
        return "Book{" +
                "bookName='" + bookName + '\'' +
                ", author='" + author + '\'' +
                '}';
    }
}

  下面创建一个Library类,由于这个Library作为被观察者,我们使其继承自抽象类AbsObservable

package demo.springboot.web;

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

/**
 * 
 * @date 2018/12/25 16:22
 */
public class Library extends AbsObservable {
    //使用list用于存放图书
    private List<Book> bookList;

    public Library() {
        // TODO Auto-generated constructor stub
        this.bookList = new ArrayList<>();
        //添加两本书
        Book ShuiHu = new Book("水浒传","施耐庵");
        Book HongLou = new Book("红楼梦", "曹雪芹");
        this.bookList.add(ShuiHu);
        this.bookList.add(HongLou);
    }


    public void addBook(Book book) {
        this.bookList.add(book);
        super.notifyAllObserver(book);
    }

    public void delBook(Book book) {
        this.bookList.remove(book);
    }

}

下面再创建两个读者类ReaderA,ReaderB。它们作为观察者,就叫他们实现Observer接口。 
 ReaderA如下

package demo.springboot.web;

/**
 * @date 2018/12/25 16:30
 */
public class ReaderA implements Observer {

    public ReaderA() {
    }

    @Override
    public void update(Object object) {
        System.out.println("我是读者A,收到了新书:" + object.toString());
    }
}

ReaderB如下

package demo.springboot.web;

/**
 * @date 2018/12/25 16:31
 */
public class ReaderB implements Observer {

    public ReaderB() {
    }

    @Override
    public void update(Object object) {
        System.out.println("我是读者B,收到了新书:" + object.toString());
    }
}

接下来,写个main函数,测试一下结果输出。

package demo.springboot.web;

/**
 * @date 2018/12/25 16:31
 */
public class TestObserver {
    public static void main(String[] args) {
        Library library = new Library();
        Observer readerAObserver = new ReaderA();
        Observer readerBObserver = new ReaderB();

        library.addObserver(readerAObserver);
        library.addObserver(readerBObserver);

        Book book = new Book("朝花夕拾", "鲁迅");
        library.addBook(book);
    }
}

输出结果如下所示:

 

观察者模式通用类图和通用代码

 首先我们看一下观察者模式的通用类图 

 

在这里对上面类图中的角色进行一下详细介绍

Subject:抽象主题,也就是上面的被观察者(Observable)角色。Subject把所有观察者对象的引用保存在一个集合里,并且能够动态的增加、取消观察者。
Observer:抽象观察者,他是观察者的抽象类,在这里定义个一个更新的接口,目的就是为了在接收到被观察者的更改通知是更新自己。
ConcreteSubject:具体主题,也就是具体的被观察者,在上面对应于Library类。在ConcreteSubject里面定义一些被观察者自己的业务逻辑,当ConcreteSubject内部状态发生改变时,给所有注册过的观察者发送通知。
ConcreteObserver:具体观察者,ConcreteObserver实现了抽象观察者所定义的更新接口,再被观察者通知改变是更新自身状态。 


下面我们就来看一下观察者模式的通用代码。 
Subject(被观察者类)

package demo.springboot.web.observer;

import java.util.Vector;

/**
 * @date 2018/12/25 17:02
 */
public abstract class Subject {

    //定义一个观察这数组
    private Vector<Observer> obVector = new Vector<>();

    //添加一个观察者
    public void addObserver(Observer observer) {
        this.obVector.add(observer);
    }

    //删除一个观察者
    public void delObserver(Observer observer) {
        this.obVector.remove(observer);
    }

    //通知所有观察者
    public void notifyObservers() {
        for (Observer observer : obVector) {
            observer.update();
        }
    }
}

ConcreteSubject(具体被观察者)

package demo.springboot.web.observer;

/**
 * @date 2018/12/25 17:04
 */
public class ConcreteSubject extends  Subject {
    // 具体业务
    public void doSomething() {
        super.notifyObservers();
    }
}

Observer(观察者类)

package demo.springboot.web.observer;

/**
 * @date 2018/12/25 17:05
 */
public interface Observer {
    public void update();
}

ConcreteObserver(具体观察者类)

package demo.springboot.web.observer;

/**
 * @date 2018/12/25 17:06
 */
public class ConcreteObserver implements Observer {
    public ConcreteObserver() {
        // TODO Auto-generated constructor stub
    }
    //实现更新方法
    @Override
    public void update() {
        // TODO Auto-generated method stub
        System.out.println("我已接收到消息");
    }
}

 Client(场景类)

package demo.springboot.web.observer;

/**
 * @date 2018/12/25 17:06
 */
public class Client {

    public static void main(String[] args) {
        // TODO Auto-generated method stub
        ConcreteSubject subject = new ConcreteSubject();
        Observer observer = new ConcreteObserver();
        subject.addObserver(observer);
        subject.doSomething();
    }
}

  Java对观察者模式提供的支持

   在java.util库里面,提供了一个Observable类和一个Observer接口。下面我们就来看一下这个Observable类和Observer接口。 
    Observer接口 
    在Observer接口中只提供了一个update方法,在被观察者发生变化时通过notifyObservers方法通知观察者做出改变,也就是执行了update方法。

package java.util;

public interface Observer {
    void update(Observable o, Object arg);
}

 Observable类 
 下面被观察者类也是为我们提供了对于观察者添加,删除,通知观察者改变等方法。当我们的需要通知观察者并且需要调用观察者update方法,我们需要调用setChanged方法。

package java.util;

public class Observable {
    private boolean changed = false;  //标记此 Observable对象为已改变的对象

    private Vector<Observer> obs;

    public Observable() {
        obs = new Vector<>();
    }

    /**
     * 添加观察者
     */
    public synchronized void addObserver(Observer o) {
        if (o == null)
            throw new NullPointerException();
        if (!obs.contains(o)) {
            obs.addElement(o);
        }
    }

    /**
     * 删除观察者
     */
    public synchronized void deleteObserver(Observer o) {
        obs.removeElement(o);
    }

    public void notifyObservers() {
        notifyObservers(null);
    }

     /**
     * 如果当前Observable对象有变化(那时hasChanged 方法会返回true)
     * 调用这个方法通知所有登记的观察者,即调用它们的update()方法
     * 传入Observer和arg作为参数
     */
    public void notifyObservers(Object arg) {

        Object[] arrLocal;

        synchronized (this) {

            if (!changed)
                return;
            arrLocal = obs.toArray();
            clearChanged();
        }

        for (int i = arrLocal.length-1; i>=0; i--)
            ((Observer)arrLocal[i]).update(this, arg);
    }

    /**
     * 删除所有的观察者
     */
    public synchronized void deleteObservers() {
        obs.removeAllElements();
    }

    /**
     *将此Observable对象标记为已改变
     */
    protected synchronized void setChanged() {
        changed = true;
    }

    /**
     *将此Observable对象标记为未改变
     */
    protected synchronized void clearChanged() {
        changed = false;
    }

    /**
     *检测当前的Observable是否发生改变
     */
    public synchronized boolean hasChanged() {
        return changed;
    }

    /**
     * 观察者的数量
     */
    public synchronized int countObservers() {
        return obs.size();
    }
}

 

  下面我们就用Java为我们提供的接口来实现我们文章开始所假定的场景。 
  Library类(注意:已标注了更改处。)

package demo.springboot.web;

import java.util.ArrayList;
import java.util.List;
//更改处:1
import java.util.Observable;

/**
 * @date 2018/12/25 16:22
 */
//更改处:2
public class Library extends Observable {
    //使用list用于存放图书
    private List<Book> bookList;

    public Library() {
        // TODO Auto-generated constructor stub
        this.bookList = new ArrayList<>();
        //添加两本书
        Book ShuiHu = new Book("水浒传","施耐庵");
        Book HongLou = new Book("红楼梦", "曹雪芹");
        this.bookList.add(ShuiHu);
        this.bookList.add(HongLou);
    }


    public void addBook(Book book) {
        //更改处3:重要
        super.setChanged();
        this.bookList.add(book);
        //更改处:4
        super.notifyObservers(book);
    }

    public void delBook(Book book) {
        this.bookList.remove(book);
    }

}

 当我们添加一本书的时候,也就是addBook方法内,我们调用了父类setChanged方法,这样才能够执行观察者的update方法。对于具体观察者Reader我们只需要实现java.util中的Observer接口即可。ReaderB与ReaderA一样

package demo.springboot.web;

import java.util.Observable;
//更改处:1
import java.util.Observer;

/**
 * @date 2018/12/25 16:30
 */
//更改处:2
public class ReaderA implements Observer {

    public ReaderA() {
    }

//    @Override
//    public void update(Object object) {
//        System.out.println("我是读者A,收到了新书:" + object.toString());
//    }

    @Override
    //更改处:3
    public void update(Observable o, Object object) {
        System.out.println("我是读者A,收到了新书:" + object.toString());
    }
}

测试上述程序,代码及结果如下所示:

package demo.springboot.web;
import java.util.Observer;

/**
 * @date 2018/12/25 16:31
 */
public class TestObserver {
    public static void main(String[] args) {
        Library library = new Library();
        Observer readerAObserver = new ReaderA();
        Observer readerBObserver = new ReaderB();

        library.addObserver(readerAObserver);
        library.addObserver(readerBObserver);

        Book book = new Book("朝花夕拾", "鲁迅");
        library.addBook(book);
        System.out.println("hahaha");
    }
}

END

注:文章转载自Java设计模式之观察者模式--无嘴小呆子

对其中的代码自己手动输入执行了一次,且更改了部分文字输入错误部分以及容易让人困惑的类名。该文讲解观察者模式清晰透彻,值得推荐!

 

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值