觀察者模式-Observer

觀察者模式很好理解鳍刷,簡單來說就是:當一個對象變化時,其它依賴該對象的對象都會收到通知幼东,并且隨著變化确镊!對象之間是一種一對多的關(guān)系士骤。

手工創(chuàng)建Observer模式

  • 創(chuàng)建觀察者接口
public interface Observer {
    void update();
}
  • 創(chuàng)建Publisher接口
public interface Publisher {
    void add(Observer observer);

    void delete(Observer observer);

    void notifyObservers();

    void operation();
}
  • 基本功能實現(xiàn)(為了線程安全我們可以選擇Vector)
import java.util.ArrayList;
import java.util.List;

public abstract class PublisherAdapter implements Publisher {
    private List<Observer> observers = new ArrayList<Observer>();

    @Override
    public void add(Observer observer) {
        observers.add(observer);
    }

    @Override
    public void delete(Observer observer) {
        observers.remove(observer);
    }

    @Override
    public void notifyObservers() {
        for (Observer observer : observers)
            observer.update();
    }

}
  • 實現(xiàn)類
public class PublisherImpl extends PublisherAdapter {

    @Override
    public void operation() {
        System.out.println("this is operation");
        notifyObservers();
    }

}
  • 測試類
public class Main {

    public static void main(String[] args) {
        Publisher impl = new PublisherImpl();
        impl.add(new Observer() {
            
            @Override
            public void update() {
                System.out.println("observer 1 receive");
            }
        });
        impl.add(new Observer() {
            
            @Override
            public void update() {
                // TODO Auto-generated method stub
                System.out.println("observer 2 receive");
            }
        });
        impl.operation();
    }

}

測試結(jié)果:

this is operation
observer 1 receive
observer 2 receive

使用java.util.Observer接口

這個接口只定義了一個方法:update()。當被觀察者(Observable)對象的狀態(tài)發(fā)生變化時骚腥,這個方法就會被調(diào)用敦间。通過調(diào)用被觀察者對象的notifyObservers()方法通知所有的觀察對象瓶逃。

  • 定義被觀察者
    被觀察者類都是java.util.Observable類的子類
import java.util.Observable;

public class Publisher extends Observable {
    public void operation() {
        System.out.println("this is operation");
        this.setChanged();
        this.notifyObservers();
    }
}
  • 測試
import java.util.Observable;
import java.util.Observer;

public class Main {

    public static void main(String[] args) {
        Publisher publisher = new Publisher();
        publisher.addObserver(new Observer() {

            @Override
            public void update(Observable o, Object arg) {
                System.out.println("observer 1 receive");
            }
        });
        publisher.addObserver(new Observer() {

            @Override
            public void update(Observable o, Object arg) {
                System.out.println("observer 2 receive");
            }
        });
        publisher.operation();
    }

}

當然束铭,若要獲得被觀察者的信息,可以將Observable o轉(zhuǎn)換為Publisher類型:
Publisher p = (Publisher)o;

分析Observer與Observable

如前所述厢绝,Observer接口只定義了一個方法:update()契沫。當被觀察者對象的狀態(tài)發(fā)生變化時,這個方法就會被調(diào)用昔汉。

java.util.Observer接口定義如下:

package java.util;

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

被觀察者類都是java.util.Observable類的子類懈万。java.util.Observable提供公開的方法支持觀察者對象,這些方法中有兩個非常重要
setChanged():被調(diào)用之后會設(shè)置一個內(nèi)部標記變量靶病,代表被觀察者對象的狀態(tài)發(fā)生了變化会通;
notifyObservers():這個方法被調(diào)用時,會調(diào)用所有登記過的觀察者對象的update()方法娄周。


附源碼

package java.util;

public class Observable {
    private boolean changed = false;
    private Vector 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);
    }

    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();
    }

    protected synchronized void setChanged() {
        changed = true;
    }

    protected synchronized void clearChanged() {
        changed = false;
    }

    public synchronized boolean hasChanged() {
        return changed;
    }

    public synchronized int countObservers() {
        return obs.size();
    }
}

可用如下類圖表示(圖片來自于網(wǎng)絡):

總結(jié)

觀察者模式優(yōu)點:

  • 觀察者模式在被觀察者和觀察者之間建立一個抽象的耦合涕侈。被觀察者角色所知道的只是一個具體觀察者列表,被觀察者和觀察者沒有緊密地耦合在一起煤辨,因此它們可以屬于不同的抽象化層次裳涛。
  • 觀察者模式支持廣播通訊。被觀察者會向所有的登記過的觀察者發(fā)出通知众辨。

觀察者模式缺點:

  • 如果一個被觀察者對象有很多的直接和間接的觀察者的話端三,將所有的觀察者都通知到會花費很多時間
  • 如果在被觀察者之間有循環(huán)依賴的話鹃彻,被觀察者會觸發(fā)它們之間進行循環(huán)調(diào)用郊闯,導致系統(tǒng)崩潰。
  • 雖然觀察者模式可以隨時使觀察者知道所觀察的對象發(fā)生了變化蛛株,但是觀察者模式?jīng)]有相應的機制使觀察者知道所觀察的對象是怎么發(fā)生變化的团赁。

代碼在這里

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末泳挥,一起剝皮案震驚了整個濱河市然痊,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌屉符,老刑警劉巖剧浸,帶你破解...
    沈念sama閱讀 222,252評論 6 516
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件锹引,死亡現(xiàn)場離奇詭異,居然都是意外死亡唆香,警方通過查閱死者的電腦和手機嫌变,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,886評論 3 399
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來躬它,“玉大人腾啥,你說我怎么就攤上這事》胂牛” “怎么了倘待?”我有些...
    開封第一講書人閱讀 168,814評論 0 361
  • 文/不壞的土叔 我叫張陵,是天一觀的道長组贺。 經(jīng)常有香客問我凸舵,道長,這世上最難降的妖魔是什么失尖? 我笑而不...
    開封第一講書人閱讀 59,869評論 1 299
  • 正文 為了忘掉前任啊奄,我火速辦了婚禮,結(jié)果婚禮上掀潮,老公的妹妹穿的比我還像新娘菇夸。我一直安慰自己,他們只是感情好仪吧,可當我...
    茶點故事閱讀 68,888評論 6 398
  • 文/花漫 我一把揭開白布庄新。 她就那樣靜靜地躺著,像睡著了一般邑商。 火紅的嫁衣襯著肌膚如雪摄咆。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,475評論 1 312
  • 那天人断,我揣著相機與錄音吭从,去河邊找鬼。 笑死恶迈,一個胖子當著我的面吹牛涩金,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播暇仲,決...
    沈念sama閱讀 41,010評論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼步做,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了奈附?” 一聲冷哼從身側(cè)響起全度,我...
    開封第一講書人閱讀 39,924評論 0 277
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎斥滤,沒想到半個月后将鸵,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體勉盅,經(jīng)...
    沈念sama閱讀 46,469評論 1 319
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,552評論 3 342
  • 正文 我和宋清朗相戀三年顶掉,在試婚紗的時候發(fā)現(xiàn)自己被綠了草娜。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,680評論 1 353
  • 序言:一個原本活蹦亂跳的男人離奇死亡痒筒,死狀恐怖宰闰,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情簿透,我是刑警寧澤移袍,帶...
    沈念sama閱讀 36,362評論 5 351
  • 正文 年R本政府宣布,位于F島的核電站萎战,受9級特大地震影響咐容,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜蚂维,卻給世界環(huán)境...
    茶點故事閱讀 42,037評論 3 335
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望路狮。 院中可真熱鬧虫啥,春花似錦、人聲如沸奄妨。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,519評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽砸抛。三九已至评雌,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間直焙,已是汗流浹背景东。 一陣腳步聲響...
    開封第一講書人閱讀 33,621評論 1 274
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留奔誓,地道東北人斤吐。 一個月前我還...
    沈念sama閱讀 49,099評論 3 378
  • 正文 我出身青樓,卻偏偏與公主長得像厨喂,于是被迫代替她去往敵國和親和措。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 45,691評論 2 361

推薦閱讀更多精彩內(nèi)容