目录
遇到问题
梳理需求
观察者模式的实现
JDK中的实现
遇到问题
当一个对象发生修改时,需要通知多方。
很自然就会想到回调,这个就是观察者模式的核心,观察者模式可以将大量的回调解耦,从而使代码更加优雅。
梳理需求
意图:定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新。
主要解决:一个对象状态改变给其他对象通知的问题,而且要考虑到易用和低耦合,保证高度的协作。
何时使用:一个对象(目标对象)的状态发生改变,所有的依赖对象(观察者对象)都将得到通知,进行广播通知。
如何解决:使用面向对象技术,可以将这种依赖关系弱化。
关键代码:在抽象类里有一个 ArrayList 存放观察者们。
观察者模式的实现
import java.util.ArrayList;import java.util.List; public class Subject {private List observers = new ArrayList(); private int state;public int getState() {return state; }public void setState(int state) {this.state = state;notifyAllObservers(); }public void attach(Observer observer){observers.add(observer); }public void notifyAllObservers(){for (Observer observer : observers) { observer.update();} }}
接下来定义它的观察者们
public abstract class Observer { protected Subject subject; public abstract void update();}
编写Observer的实现类
public class BinaryObserver extends Observer{public BinaryObserver(Subject subject){this.subject = subject;this.subject.attach(this); }@Override public void update() {System.out.println( "Binary String: " + Integer.toBinaryString( subject.getState() ) );}}
public class OctalObserver extends Observer{public OctalObserver(Subject subject){this.subject = subject;this.subject.attach(this); }@Override public void update() { System.out.println( "Octal String: "+ Integer.toOctalString( subject.getState() ) );}}
public class HexaObserver extends Observer{public HexaObserver(Subject subject){this.subject = subject;this.subject.attach(this); }@Override public void update() {System.out.println( "Hex String: " + Integer.toHexString( subject.getState() ).toUpperCase() );}}
使用
public class ObserverPatternDemo { public static void main(String[] args) {Subject subject = new Subject(); new HexaObserver(subject);new OctalObserver(subject);new BinaryObserver(subject); System.out.println("First state change: 15"); subject.setState(15);System.out.println("Second state change: 10");subject.setState(10); }}
JDK中的实现
Java的JDK中,已经实现观察者模式的相关代码
JDK中提供了Observable抽象类、以及Observer接口
Observable:可被观察的
主要包含两个方法:
setChanged()
每次状态发生变更,都需要手动调用setChanged
如果需要通知观察者,则需要调用
notifyObservers()
注意:如果在调用notifyObservers()方法之前没有调用setChanged()方法,就不会有什么动作发生。
notifyObservers()方法中包含clearChanged()方法,将标志变量置回原值。
notifyObservers()方法采用的是从后向前的遍历方式,即最后加入的观察者最先被调用update()方法。
package com.learnjava.observer;import java.util.Observable;import java.util.Observer;class WatchedCounter extends Observable{public void countdown(int number){for (; number >= 0; --number){// 设置改变变量setChanged();// 通知所有观察者,将number作为参数信息传递给观察者notifyObservers(number);}}}class Watcher1 implements Observer{@Overridepublic void update(Observable arg0, Object arg1){System.out.println("Watcher1's number: " + arg1);}}class Watcher2 implements Observer{@Overridepublic void update(Observable arg0, Object arg1){if (((Integer) arg1).intValue() <= 5){System.out.println("Watcher2's number: " + arg1);}}}public class ObserverTest{public static void main(String[] args){WatchedCounter watchedCounter = new WatchedCounter();Watcher1 watcher1 = new Watcher1();Watcher2 watcher2 = new Watcher2();//添加观察者watchedCounter.addObserver(watcher1);watchedCounter.addObserver(watcher2);//开始倒数计数watchedCounter.countdown(10);}}