1、使用場景
當對象間存在一對多關係時,則使用觀察者模式。當一個對象被修改時,則會通知他的依賴對象,並進一步引起多個其他對象對此對象的行爲作出反應。
場景1:新聞評論模塊
當用戶發佈評論時,會在評論展示模塊末尾處追加新的評論,同時用戶的消息模塊數量也會遞增。刪除信息時相反。
場景2:訂閱功能(發佈-訂閱模式)
如微博的訂閱功能,當我們訂閱了某人的微博賬號,此人發佈新的消息時,就會通知我們。
場景3:數據變化通知
如vue中的mvvm模式,當視圖發生變化時,會引起數據的同步變化。
場景4:觸發鏈
A的行爲觸發B的某個行爲,B的行爲觸發C的行爲...依據此模式創建鏈式行爲。
2、對象劃分及功能作用
1、Subject(被觀察者)
Subject對象具有將觀察者綁定到Client的方法、從Client對象解綁觀察者的方法。它的作用是管理觀察關係、產生特定行爲後通知並調用觀察者的行爲。
2、Observer(觀察者,一般有多個對象)
Observer一般具有一個抽象接口和多個具體觀察者(實現類),每個類代表影響到的一個行爲。
3、Client(使用者)
從調用者的角度看如何使用觀察者模式、使用的效果。
例:
被觀察者:
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();
}
}
}
觀察者:
public abstract class Observer {
protected Subject subject;
public abstract void update();
}
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);
}
}