設(shè)計模式走一遍---觀察者模式

1

紅燈車過宗侦,人停愚臀;綠燈人過,車停矾利。每天走在馬路上姑裂,到處可見紅綠燈指揮著我們什么時候可以過馬路馋袜,什么時候不能過馬路。無論是人還是車舶斧,都時刻關(guān)注著紅綠燈的狀態(tài)欣鳖,一旦紅綠燈的狀態(tài)發(fā)生了改變,我們總能第一時間發(fā)現(xiàn)茴厉,并且做出相應(yīng)的響應(yīng).....說真观堂,紅綠燈真的是個偉大的發(fā)明。

說到觀察者模式呀忧,無非就是觀察者被觀察者之間牽扯到的一些關(guān)系师痕。在上面的紅綠燈例子中,紅綠燈就如同被觀察者而账,我們又稱之為觀察目標(biāo)胰坟,而人行者或開著車的人就如同觀察者,時刻觀察著紅綠燈的變化泞辐,紅綠燈一旦發(fā)生變化笔横,便會馬上通知觀察者,觀察者也經(jīng)常會做出相應(yīng)的反應(yīng)咐吼。

下面我們說下觀察者模式的定義:

觀察者模式定義了對象之間的一種一對多依賴關(guān)系吹缔,使得每當(dāng)一個對象狀態(tài)發(fā)生改變時,其相關(guān)依賴對象皆得到通知并被自動更新锯茄。

觀察者模式的別名包括發(fā)布-訂閱(Publish/Subscribe)模式厢塘、模型-視圖(Model/View)模式、源-監(jiān)聽器(Source/Listener)模式或從屬者(Dependents)模式肌幽。

上面的例子中晚碾,紅綠燈就相當(dāng)于,而路上的人就相當(dāng)于喂急,每次紅路燈這個目標(biāo)對象的狀態(tài)發(fā)生變化格嘁,就會通知眾多的觀察者(人)。而觀察者一般也會做出對象的響應(yīng)

觀察者模式屬于行為型模式

2

觀察者模式主要解決的問題:一方的狀態(tài)發(fā)生了變化廊移,依賴于這一方的觀察者立即能收到通知糕簿。

例如我們平時訂閱的微信公眾號,一旦公眾號有新的文章發(fā)布狡孔,訂閱者能夠立即收到新的文章推送懂诗。

這里需要注意的是,目標(biāo)對象會把狀態(tài)的變化通知所有觀察者步氏,而不管觀察者的具體身份响禽。自己也并不知道通知的這個人究竟是誰徒爹。

3

觀察者模式一般包含如下四個角色:

  1. Subject:目標(biāo)對象荚醒,一般設(shè)計成抽象類
  2. ConcreteSubject:具體目標(biāo)對象芋类,Subject的子類。
  3. Observer:觀察者界阁,一般設(shè)計為接口
  4. ConcreteObserver:具體觀察者侯繁,Observer的實現(xiàn)者

結(jié)構(gòu)圖:

image

下面具體介紹下這四個角色:

Subject(目標(biāo)):目標(biāo)又被稱為主題,指被觀察的對象泡躯,即被觀察者贮竟。一般我們會在在目標(biāo)中定義一個觀察者集合,用來管理觀察者较剃。一個觀察目標(biāo)可以接受任意數(shù)量的觀察者來觀察咕别,它提供一系列方法來增加和刪除觀察者對象,如attach()方法與detach()方法写穴;同時也會定義通知方法notify()惰拱。目標(biāo)類可以是接口,也可以是抽象類或具體類啊送,但一般我們設(shè)計為抽象類偿短。

ConcreteSubject(具體目標(biāo)):具體目標(biāo)是目標(biāo)類的子類(接口的實現(xiàn)者),通常它包含有經(jīng)常發(fā)生改變的數(shù)據(jù)馋没,當(dāng)它的狀態(tài)發(fā)生改變時昔逗,向它的各個觀察者發(fā)出通知;同時它還實現(xiàn)了在目標(biāo)類中定義的抽象業(yè)務(wù)邏輯方法

Observer(觀察者):觀察者角色一般是一個接口篷朵,它會有一個update方法勾怒,當(dāng)目標(biāo)對象的狀態(tài)發(fā)生改變時,這個方法就會被調(diào)用声旺。

ConcreteObserver(具體觀察者):觀察者接口的實現(xiàn)者控硼,在這個角色中,將會定義目標(biāo)對象狀態(tài)發(fā)生變化時所要處理的邏輯艾少。

觀察者模式一般的代碼實現(xiàn):

1.目標(biāo)對象與具體目標(biāo)對象代碼示例

public abstract class Subject {  
    //定義一個觀察者集合用于存儲所有觀察者對象  
protected List<Observer> observers = new ArrayList();  

    //注冊方法卡乾,用于向觀察者集合中增加一個觀察者  
    public void attach(Observer observer) {  
    observers.add(observer);  
}  

    //注銷方法,用于在觀察者集合中刪除一個觀察者  
    public void detach(Observer observer) {  
    observers.remove(observer);  
}  

    //聲明抽象通知方法  
    public abstract void notify();  
    
    //其他方法
}  


//具體目標(biāo)類ConcreteSubject是實現(xiàn)了抽象目標(biāo)類Subject的一個具體子類
//其典型代碼如下所示:

class ConcreteSubject extends Subject {  

    //實現(xiàn)通知方法  
    public void notify() {  
    System.out.println("目標(biāo)對象狀態(tài)發(fā)生變化了")
    //遍歷觀察者集合缚够,調(diào)用每一個觀察者的響應(yīng)方法  
        for(Observer obs:observers) {  
            obs.update();  
        }  
    }     
}

2.觀察者與具體觀察者代碼示例

interface Observer {  
    //聲明響應(yīng)方法  
    public void update();  
}  

//在具體觀察者ConcreteObserver中實現(xiàn)了update()方法
//其典型代碼如下所示:
class ConcreteObserver implements Observer {  
    //實現(xiàn)響應(yīng)方法  
    public void update() {  
        System.out.println("觀察者收到通知幔妨,正在做相應(yīng)的處理")  
    }  
}

3.測試代碼

public class Test{
    public static void main(String[] args){
        Subject sub = new ConcreteSubject();
        sub.attach(new ConcreteObserver());
        //假設(shè)狀態(tài)發(fā)生了變化調(diào)用notify()方法
        sub.notify();
    }
}

4.打印結(jié)果

目標(biāo)對象狀態(tài)發(fā)生變化了
觀察者收到通知,正在做相應(yīng)的處理

4

優(yōu)點:

1谍椅、從例子中我們可以看出觀察者和被觀察者是抽象耦合的误堡,只有輕微的關(guān)聯(lián)關(guān)系

2、建立一套觸發(fā)機制雏吭。目標(biāo)對象一旦發(fā)生變化锁施,便會觸發(fā)廣播通知,觀察者一旦收到通知,也會觸發(fā)相應(yīng)的響應(yīng)悉抵。

缺點:

1肩狂、如果一個被觀察者對象有很多的直接和間接的觀察者的話,將所有的觀察者都通知到會花費很多時間姥饰。

2傻谁、如果在觀察者和觀察目標(biāo)之間有循環(huán)依賴的話,觀察目標(biāo)會觸發(fā)它們之間進(jìn)行循環(huán)調(diào)用列粪,可能導(dǎo)致系統(tǒng)崩潰审磁。

3、觀察者模式?jīng)]有相應(yīng)的機制讓觀察者知道所觀察的目標(biāo)對象是怎么發(fā)生變化的岂座,而僅僅只是知道觀察目標(biāo)發(fā)生了變化态蒂。

使用場景:

1.一個抽象模型有兩個方面,其中一個方面依賴于另一個方面费什。將這些方面封裝在獨立的對象中使它們可以各自獨立地改變和復(fù)用吃媒。

2.一個對象的改變將導(dǎo)致其他一個或多個對象也發(fā)生改變,而不知道具體有多少對象將發(fā)生改變吕喘,可以降低對象之間的耦合度赘那。

3.一個對象必須通知其他對象,而并不知道這些對象是誰氯质。

4.需要在系統(tǒng)中創(chuàng)建一個觸發(fā)鏈募舟,A對象的行為將影響B(tài)對象,B對象的行為將影響C對象……闻察,可以使用觀察者模式創(chuàng)建一種鏈?zhǔn)接|發(fā)機制

最后

Java語言中也有提供了Observer接口拱礁,下一篇簡單講解使用下。

參考書籍:

1.設(shè)計模式j(luò)ava版辕漂。

2.Head First設(shè)計模式

3.菜鳥教程網(wǎng)站

關(guān)注公我的眾號:苦逼的碼農(nóng)呢灶,獲取更多原創(chuàng)文章,后臺回復(fù)禮包送你一份特別的資源大禮包钉嘹。同時也感謝把文章介紹給更多需要的人

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末鸯乃,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子跋涣,更是在濱河造成了極大的恐慌缨睡,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,509評論 6 504
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件陈辱,死亡現(xiàn)場離奇詭異奖年,居然都是意外死亡,警方通過查閱死者的電腦和手機沛贪,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,806評論 3 394
  • 文/潘曉璐 我一進(jìn)店門陋守,熙熙樓的掌柜王于貴愁眉苦臉地迎上來震贵,“玉大人,你說我怎么就攤上這事水评⌒上担” “怎么了?”我有些...
    開封第一講書人閱讀 163,875評論 0 354
  • 文/不壞的土叔 我叫張陵之碗,是天一觀的道長蝙眶。 經(jīng)常有香客問我季希,道長褪那,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,441評論 1 293
  • 正文 為了忘掉前任式塌,我火速辦了婚禮博敬,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘峰尝。我一直安慰自己偏窝,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 67,488評論 6 392
  • 文/花漫 我一把揭開白布武学。 她就那樣靜靜地躺著祭往,像睡著了一般。 火紅的嫁衣襯著肌膚如雪火窒。 梳的紋絲不亂的頭發(fā)上硼补,一...
    開封第一講書人閱讀 51,365評論 1 302
  • 那天,我揣著相機與錄音熏矿,去河邊找鬼已骇。 笑死,一個胖子當(dāng)著我的面吹牛票编,可吹牛的內(nèi)容都是我干的褪储。 我是一名探鬼主播,決...
    沈念sama閱讀 40,190評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼慧域,長吁一口氣:“原來是場噩夢啊……” “哼鲤竹!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起昔榴,我...
    開封第一講書人閱讀 39,062評論 0 276
  • 序言:老撾萬榮一對情侶失蹤宛裕,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后论泛,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體揩尸,經(jīng)...
    沈念sama閱讀 45,500評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,706評論 3 335
  • 正文 我和宋清朗相戀三年屁奏,在試婚紗的時候發(fā)現(xiàn)自己被綠了岩榆。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,834評論 1 347
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖勇边,靈堂內(nèi)的尸體忽然破棺而出犹撒,到底是詐尸還是另有隱情,我是刑警寧澤粒褒,帶...
    沈念sama閱讀 35,559評論 5 345
  • 正文 年R本政府宣布识颊,位于F島的核電站,受9級特大地震影響奕坟,放射性物質(zhì)發(fā)生泄漏祥款。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,167評論 3 328
  • 文/蒙蒙 一月杉、第九天 我趴在偏房一處隱蔽的房頂上張望刃跛。 院中可真熱鬧,春花似錦苛萎、人聲如沸桨昙。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,779評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽蛙酪。三九已至,卻和暖如春翘盖,著一層夾襖步出監(jiān)牢的瞬間桂塞,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,912評論 1 269
  • 我被黑心中介騙來泰國打工最仑, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留藐俺,地道東北人。 一個月前我還...
    沈念sama閱讀 47,958評論 2 370
  • 正文 我出身青樓泥彤,卻偏偏與公主長得像欲芹,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子吟吝,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,779評論 2 354

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

  • 設(shè)計模式概述 在學(xué)習(xí)面向?qū)ο笃叽笤O(shè)計原則時需要注意以下幾點:a) 高內(nèi)聚菱父、低耦合和單一職能的“沖突”實際上,這兩者...
    彥幀閱讀 3,743評論 0 14
  • 1 場景問題# 1.1 訂閱報紙的過程## 來考慮實際生活中訂閱報紙的過程剑逃,這里簡單總結(jié)了一下浙宜,訂閱報紙的基本流程...
    七寸知架構(gòu)閱讀 4,621評論 5 57
  • 【學(xué)習(xí)難度:★★★☆☆,使用頻率:★★★★★】直接出處:觀察者模式梳理和學(xué)習(xí):https://github.com...
    BruceOuyang閱讀 1,526評論 1 5
  • 參考資料:菜鳥教程之設(shè)計模式 設(shè)計模式概述 設(shè)計模式(Design pattern)代表了最佳的實踐蛹磺,通常被有經(jīng)驗...
    Steven1997閱讀 1,173評論 1 12
  • 領(lǐng)導(dǎo)力是一種能讓別人追隨你的能力粟瞬,即使別人只是處于好奇。 在特別嚴(yán)峻的形勢下萤捆,當(dāng)“事實”似乎已經(jīng)注定了某一結(jié)果時裙品,...
    殘劍閱讀 470評論 0 0