觀察者(Observer)

意圖

定義對(duì)象間的一種一對(duì)多的依賴(lài)關(guān)系眼刃,當(dāng)一個(gè)對(duì)象的狀態(tài)發(fā)生改變時(shí),所有依賴(lài)它的對(duì)象都得到通知并自動(dòng)更新。

結(jié)構(gòu)

觀察者結(jié)構(gòu)圖
觀察者時(shí)序圖

動(dòng)機(jī)

將一個(gè)系統(tǒng)分割成一系列相互協(xié)作的類(lèi)有一個(gè)常見(jiàn)的副作用:需要維護(hù)相關(guān)對(duì)象間的一致性参淹。我們不希望為了維持一致性而使各類(lèi)緊密耦合,因?yàn)檫@樣降低了它們的可重用性乏悄。

適用性

  • 當(dāng)一個(gè)抽象模型有兩個(gè)方面, 其中一個(gè)方面依賴(lài)于另一方面浙值。將這二者封裝在獨(dú)立的對(duì)象中以使它們可以各自獨(dú)立地改變和復(fù)用;
  • 當(dāng)對(duì)一個(gè)對(duì)象的改變需要同時(shí)改變其它對(duì)象, 而不知道具體有多少對(duì)象有待改變檩小;
  • 當(dāng)一個(gè)對(duì)象必須通知其它對(duì)象开呐,而它又不能假定其它對(duì)象是誰(shuí)。換言之, 你不希望這些對(duì)象是緊密耦合的。

優(yōu)缺點(diǎn)

  • 目標(biāo)和觀察者間的抽象耦合(接口)负蚊。目標(biāo)只知道有一系列的觀察者神妹,但不知道它們所屬的具體類(lèi)颓哮;
  • 支持廣播通信家妆。目標(biāo)(Subject)發(fā)送的通知被自動(dòng)廣播給所有已向該目標(biāo)登記的所有對(duì)象(Observer);
  • 意外的更新冕茅。如果一個(gè)觀察者(Observer)誤操作目標(biāo)(Subject)的狀態(tài)伤极,可能會(huì)導(dǎo)致其他觀察者連鎖反應(yīng)式的錯(cuò)誤更新。

注意事項(xiàng)

  • 當(dāng)觀察者(Observer)依賴(lài)多個(gè)目標(biāo)(Subject)時(shí)姨伤,考慮擴(kuò)展Update接口哨坪,把目標(biāo)對(duì)象(Subject)作為識(shí)別參數(shù)。
// Subject class
public void Notify()
{
    foreah(Observer o in Observers)
    {
          o.Update(this);
    }
}
// ConcreteObserver class
public void Update(Subject subject)
{
      if(subject is ConcreteSubject1)
      {
          // 來(lái)自目標(biāo)1的通知
      }
      else if(subject is ConcreteSubject2)
      {
          // 來(lái)自目標(biāo)2的通知
      }
      ...
}
  • 在通知機(jī)制的實(shí)現(xiàn)上乍楚,可以由目標(biāo)(Subject)對(duì)象自動(dòng)觸發(fā)当编,或者由客戶(hù)端手動(dòng)觸發(fā)。
//  示例:狀態(tài)變更徒溪,目標(biāo)(Subject)自動(dòng)觸發(fā)忿偷。
public class ConcreteSubject : Subject
{
      private object state;
      public object SetState(object state)
      {
          this.state = state;
          this.Notify();   // 自動(dòng)觸發(fā)
      }

      public void Notify()
      {
            foreach(Observer o in Observers)
            {
                  o.Update();
            }
      }
}
// 示例
public class App
{
    public static void Main(string[] args)
    {
          ConcreteSubject subject = new ConcreteSubject();
          subject.Attach(Observer); // 登記觀察者
          ...
          // 狀態(tài)每一次變更,都會(huì)自動(dòng)通知觀察者
          subject.SetState(newState1); 
          subject.SetState(newState2); 
          ...
    }
}
// 示例:客戶(hù)端手動(dòng)觸發(fā)通知
public class ConcreteSubject
{
      private object state;
      public object SetState(object state)
      {
          this.state = state;
      }

      public void Notify()
      {
            foreach(Observer o in Observers)
            {
                  o.Update();
            }
      }
}
// 示例
public class App
{
    public static void Main(string[] args)
    {
          ConcreteSubject subject = new ConcreteSubject();
          subject.Attach(Observer); // 登記觀察者
          ...
          // 設(shè)置完一系列狀態(tài)后臊泌,一次性通知觀察者(避免觀察者繁瑣更新)鲤桥。
          subject.SetState(newState1); 
          subject.SetState(newState2); 
          ...
          subject.Notify();  // 手動(dòng)通知(容易遺忘)
    }
}
  • 確保目標(biāo)(Subject)在觸發(fā)通知之前,處于一致?tīng)顟B(tài)渠概。特別是在子類(lèi)集成Subject時(shí)容易發(fā)生茶凳,可用模板(Template)模式實(shí)現(xiàn);
// 存在狀態(tài)不一致的錯(cuò)誤代碼
public class ConcreteSubject : Subject
{
    ...
    public override void SetState(object state)
    {
        base.Notify();   // 提前觸發(fā)播揪,導(dǎo)致?tīng)顟B(tài)不一致贮喧。
        this.state = state;
    }
}

更改為模板實(shí)現(xiàn)方式:

public class Subject
{
    ...
    // 模板方法
    public void SetState(object state)
    {        
        this.InternalSetState(state);
        this.Notify();
    }
    protected virtual void InternalSetState(object state)
    {
        this.state = state;
    }
}
public class ConcreteSubject : Subject
{
    ...
    // 子類(lèi)只需要實(shí)現(xiàn)個(gè)性化的狀態(tài)處理
    protected override void InternalSetState(object state)
    {
        ...
        this.state = state;
    }
}
  • 可以擴(kuò)展目標(biāo)的注冊(cè)接口,讓各觀察者注冊(cè)為僅對(duì)特定事件感興趣猪狈,以提高更新的效率塞淹。
public void Attach(Observer observer, InterestType interest);


示例

模擬兩個(gè)不同類(lèi)型的圖形控件罪裹,分別顯示當(dāng)前的時(shí)間饱普。

實(shí)現(xiàn)(C#)

示例結(jié)構(gòu)圖
using System;
using System.Threading;
using System.Collections.Generic;

// 目標(biāo)主題基類(lèi)
public class Subject
{
    private readonly List<Observer> observers = new List<Observer>();

    public void Attach(Observer o)
    {
        this.observers.Add(o);
    }

    public void Detach(Observer o)
    {
        this.observers.Remove(o);
    }

    public void Notify()
    {
        foreach(Observer o in this.observers)
        {
            o.Update(this);
        }
    }
}

// 觀察者
public abstract class Observer
{
    public abstract void Update(Subject theChangedSubject);
}

// 具體的目標(biāo)主題逮诲,以3秒間隔發(fā)出通知
public class ClockTimer : Subject
{
    private Timer timer;

    public ClockTimer()
    {
        this.timer = new Timer(this.Tick, null, 0, 3000);
    }

    public void Tick(object state)
    {
        this.Now = DateTime.Now;
        this.Notify();
    }

    public DateTime Now { get; private set;}
}

// 模擬時(shí)鐘控件1
public class DigitalClock : Observer
{
    private readonly ClockTimer subject;

    public DigitalClock(ClockTimer subject)
    {
        this.subject = subject;
        this.subject.Attach(this);  // 注冊(cè)監(jiān)聽(tīng)
    }

    public override void Update(Subject theChangedSubject)
    {
        // 確認(rèn)是否為目標(biāo)監(jiān)聽(tīng)對(duì)象
        if(this.subject == theChangedSubject)
        {
            Console.WriteLine("1.DigitalClock : " + this.subject.Now);
        }
    }
}

// 模擬時(shí)鐘控件2
public class AnalogClock : Observer
{
    private readonly ClockTimer subject;

    public AnalogClock(ClockTimer subject)
    {
        this.subject = subject;
        this.subject.Attach(this); // 注冊(cè)監(jiān)聽(tīng)
    }

    public override void Update(Subject theChangedSubject)
    {
        // 確認(rèn)是否為目標(biāo)監(jiān)聽(tīng)對(duì)象
        if(this.subject == theChangedSubject)
        {
            Console.WriteLine("2. AnalogClock : " + this.subject.Now);
        }
    }
}

public class App
{
    public static void Main(string[] args)
    {
        ClockTimer timer = new ClockTimer();
        DigitalClock digitalClock = new DigitalClock(timer);
        AnalogClock analogClock = new AnalogClock(timer);

        Console.WriteLine("please enter any key to exit..\n");
        Console.Read();
    }
}

// 控制臺(tái)輸出:
//  please enter any key to exit..
//  1.DigitalClock : 2017/6/17 22:37:24
//  2. AnalogClock : 2017/6/17 22:37:24
//  1.DigitalClock : 2017/6/17 22:37:27
//  2. AnalogClock : 2017/6/17 22:37:27
//  1.DigitalClock : 2017/6/17 22:37:30
//  2. AnalogClock : 2017/6/17 22:37:30
//  1.DigitalClock : 2017/6/17 22:37:33
//  2. AnalogClock : 2017/6/17 22:37:33
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末知残,一起剝皮案震驚了整個(gè)濱河市消请,隨后出現(xiàn)的幾起案子嘱根,更是在濱河造成了極大的恐慌辩蛋,老刑警劉巖增拥,帶你破解...
    沈念sama閱讀 222,946評(píng)論 6 518
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件喜颁,死亡現(xiàn)場(chǎng)離奇詭異苍鲜,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)康愤,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,336評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門(mén)儡循,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人征冷,你說(shuō)我怎么就攤上這事择膝。” “怎么了检激?”我有些...
    開(kāi)封第一講書(shū)人閱讀 169,716評(píng)論 0 364
  • 文/不壞的土叔 我叫張陵肴捉,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我叔收,道長(zhǎng)齿穗,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 60,222評(píng)論 1 300
  • 正文 為了忘掉前任饺律,我火速辦了婚禮窃页,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘复濒。我一直安慰自己脖卖,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 69,223評(píng)論 6 398
  • 文/花漫 我一把揭開(kāi)白布芝薇。 她就那樣靜靜地躺著胚嘲,像睡著了一般。 火紅的嫁衣襯著肌膚如雪洛二。 梳的紋絲不亂的頭發(fā)上馋劈,一...
    開(kāi)封第一講書(shū)人閱讀 52,807評(píng)論 1 314
  • 那天,我揣著相機(jī)與錄音晾嘶,去河邊找鬼妓雾。 笑死,一個(gè)胖子當(dāng)著我的面吹牛垒迂,可吹牛的內(nèi)容都是我干的械姻。 我是一名探鬼主播,決...
    沈念sama閱讀 41,235評(píng)論 3 424
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼机断,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼楷拳!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起吏奸,我...
    開(kāi)封第一講書(shū)人閱讀 40,189評(píng)論 0 277
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤欢揖,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后奋蔚,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體她混,經(jīng)...
    沈念sama閱讀 46,712評(píng)論 1 320
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡烈钞,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,775評(píng)論 3 343
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了坤按。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片毯欣。...
    茶點(diǎn)故事閱讀 40,926評(píng)論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖臭脓,靈堂內(nèi)的尸體忽然破棺而出酗钞,到底是詐尸還是另有隱情,我是刑警寧澤谢鹊,帶...
    沈念sama閱讀 36,580評(píng)論 5 351
  • 正文 年R本政府宣布算吩,位于F島的核電站留凭,受9級(jí)特大地震影響佃扼,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜蔼夜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,259評(píng)論 3 336
  • 文/蒙蒙 一兼耀、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧求冷,春花似錦瘤运、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,750評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至韭山,卻和暖如春郁季,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背钱磅。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,867評(píng)論 1 274
  • 我被黑心中介騙來(lái)泰國(guó)打工梦裂, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人盖淡。 一個(gè)月前我還...
    沈念sama閱讀 49,368評(píng)論 3 379
  • 正文 我出身青樓年柠,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親褪迟。 傳聞我的和親對(duì)象是個(gè)殘疾皇子冗恨,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,930評(píng)論 2 361

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

  • 1 場(chǎng)景問(wèn)題# 1.1 訂閱報(bào)紙的過(guò)程## 來(lái)考慮實(shí)際生活中訂閱報(bào)紙的過(guò)程,這里簡(jiǎn)單總結(jié)了一下味赃,訂閱報(bào)紙的基本流程...
    七寸知架構(gòu)閱讀 4,633評(píng)論 5 57
  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語(yǔ)法掀抹,類(lèi)相關(guān)的語(yǔ)法,內(nèi)部類(lèi)的語(yǔ)法洁桌,繼承相關(guān)的語(yǔ)法渴丸,異常的語(yǔ)法,線程的語(yǔ)...
    子非魚(yú)_t_閱讀 31,669評(píng)論 18 399
  • 面向?qū)ο蟮牧笤瓌t 單一職責(zé)原則 所謂職責(zé)是指類(lèi)變化的原因。如果一個(gè)類(lèi)有多于一個(gè)的動(dòng)機(jī)被改變谱轨,那么這個(gè)類(lèi)就具有多于...
    JxMY閱讀 948評(píng)論 1 3
  • 一 引言 RxJava2.0中出現(xiàn)了兩種觀察者模式:● Observable(被觀察者)/Observer(觀察者...
    Calllanna閱讀 4,274評(píng)論 0 2
  • 我的小仙女好不容易來(lái)看我一次戒幔,還讓我給整感冒了,總覺(jué)得心里不是個(gè)滋味土童,本來(lái)你感冒了就不開(kāi)心诗茎,吃飯和回來(lái)的時(shí)候還讓我...
    寄柔閱讀 662評(píng)論 0 0