定义
当对象之间存在一对多关系时使用观察者模式,例如如果一个对象被修改,它的依赖对象将被自动通知,并且对所有依赖对象进行相应的更改。
例子
比方说,你的永久地址改变了,你需要通知护照当局和泛卡当局。所以这里护照局和泛卡局是观察者,你是主体。
同样在 Facebook 上,如果您订阅了某人,那么每当有新的更新发生时,您都会收到通知。
何时使用它:
当一个对象改变其状态时,所有其他依赖对象必须自动改变其状态以保持一致性
当对象不知道它拥有的观察者数量时。
当一个对象应该能够在不知道对象是谁的情况下通知其他对象。
第1步
创建主题类。
主题.java
import java.util.ArrayList;
import java.util.List;
public class Subject {
private List<Observer> observers
= new ArrayList<Observer>();
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();
}
}
}
第2步
创建观察者类。
观察者.java
public abstract class Observer {
protected Subject subject;
public abstract void update();
}
第 3 步
创建具体的观察者类
BinaryObserver.java
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() ) );
}
}
OctalObserver.java
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() ) );
}
}
HexaObserver.java
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() );
}
}
第4步
使用主题和具体的观察者对象。
ObserverPatternDemo.java
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);
}
}
第 5 步
验证输出。
第一次状态变化:15
十六进制字符串:F
八进制字符串:17
二进制字符串:1111
第二次状态变化:10
十六进制字符串:A
八进制字符串:12
二进制字符串:1010