觀察者模式和發(fā)布訂閱模式的區(qū)別

1.模式差異

  • 在觀察者模式中幔嗦,觀察者是知道Subject的虎敦,Subject一直保持對(duì)觀察者進(jìn)行記錄游岳。然而政敢,在發(fā)布訂閱模式中,發(fā)布者和訂閱者不知道對(duì)方的存在胚迫。它們只有通過消息代理進(jìn)行通信喷户。
  • 在發(fā)布訂閱模式中,組件是松散耦合的晌区,正好和觀察者模式相反摩骨。
  • 觀察者模式大多數(shù)時(shí)候是同步的,比如當(dāng)事件觸發(fā)朗若,Subject就會(huì)去調(diào)用觀察者的方法恼五。而發(fā)布-訂閱模式大多數(shù)時(shí)候是異步的(使用消息隊(duì)列)。
  • 觀察者 模式需要在單個(gè)應(yīng)用程序地址空間中實(shí)現(xiàn)哭懈,而發(fā)布-訂閱更像交叉應(yīng)用模式灾馒。


    觀察者模式和發(fā)布訂閱模式

2、代碼實(shí)現(xiàn)觀察者模式和發(fā)布訂閱模式

1.1 觀察者模式:
// 觀察者
class Observer {
    constructor() {

    }
    update(val) {
        console.log(val);
    }
}
// 觀察者列表
class ObserverList {
    constructor() {
        this.observerList = []
    }
    add(observer) {
        return this.observerList.push(observer);
    }
    remove(observer) {
        this.observerList = this.observerList.filter(ob => ob !== observer);
    }
    count() {
        return this.observerList.length;
    }
    get(index) {
        return this.observerList[index];
    }
}
// 目標(biāo)
class Subject {
    constructor() {
        this.observers = new ObserverList();
    }
    addObserver(observer) {
        this.observers.add(observer);
    }
    removeObserver(observer) {
        this.observers.remove(observer);
    }
    notify(...args) {
        let obCount = this.observers.count();
        for (let index = 0; index < obCount; index++) {
            this.observers.get(index).update(...args);
        }
    }
}
const subject = new Subject();
const ob1 = new Observer();
const ob2 = new Observer();
subject.addObserver(ob1);
subject.addObserver(ob2);
subject.notify(1234); // 輸出兩次 1234 

使用subject.notify()方法的時(shí)候會(huì)調(diào)度所有觀察者遣总,本例中有兩個(gè)觀察者ob1睬罗、ob2,所以會(huì)輸出兩次1234


Subject實(shí)例

觀察模式圖解:


觀察模式圖解
1.2 發(fā)布訂閱模式:
class PubSub {
    constructor() {
        this.subscribers = {}
    }
    subscribe(type, fn) {
        if (!Object.prototype.hasOwnProperty.call(this.subscribers, type)) {
          this.subscribers[type] = [];
        }
        
        this.subscribers[type].push(fn);
    }
    unsubscribe(type, fn) {
        let listeners = this.subscribers[type];
        if (!listeners || !listeners.length) return;
        const index = listeners.indexOf(fn);
        this.subscribers[type] = listeners.splice(index, 1);
    }
    publish(type, ...args) {
        let listeners = this.subscribers[type];
        if (!listeners || !listeners.length) return;
        listeners.forEach(fn => fn(...args));        
    }
}
let ob = new PubSub();
function oneFunction(val){console.log('one參數(shù)'+val)}
function twoFunction(val){console.log('two參數(shù)'+val)}
ob.subscribe('one',oneFunction);
ob.subscribe('two',twoFunction);
ob.publish('one',1); // one參數(shù)1
ob.publish('two',2); // two參數(shù)2

在「一對(duì)多」的場(chǎng)景下旭斥,發(fā)布者更新可以通知它的部分訂閱者容达,例如本例中one和two可以分開訂閱消息,不像觀察者模式主體(subject)發(fā)生變化所有的觀察者(ob1和ob2)都得到通知垂券。


發(fā)布訂閱模式圖解

參考文章:
觀察者模式和發(fā)布訂閱模式的區(qū)別
觀察者模式(Observer)和發(fā)布(Publish)/訂閱模式(Subscribe)的區(qū)別
設(shè)計(jì)模式之發(fā)布訂閱模式(1) 一文搞懂發(fā)布訂閱模式

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末花盐,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子菇爪,更是在濱河造成了極大的恐慌算芯,老刑警劉巖,帶你破解...
    沈念sama閱讀 212,454評(píng)論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件凳宙,死亡現(xiàn)場(chǎng)離奇詭異熙揍,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)氏涩,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,553評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門届囚,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人是尖,你說我怎么就攤上這事奖亚。” “怎么了析砸?”我有些...
    開封第一講書人閱讀 157,921評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵昔字,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我,道長(zhǎng)作郭,這世上最難降的妖魔是什么陨囊? 我笑而不...
    開封第一講書人閱讀 56,648評(píng)論 1 284
  • 正文 為了忘掉前任,我火速辦了婚禮夹攒,結(jié)果婚禮上蜘醋,老公的妹妹穿的比我還像新娘。我一直安慰自己咏尝,他們只是感情好压语,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,770評(píng)論 6 386
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著编检,像睡著了一般胎食。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上允懂,一...
    開封第一講書人閱讀 49,950評(píng)論 1 291
  • 那天厕怜,我揣著相機(jī)與錄音,去河邊找鬼蕾总。 笑死粥航,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的生百。 我是一名探鬼主播递雀,決...
    沈念sama閱讀 39,090評(píng)論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼蚀浆!你這毒婦竟也來了缀程?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,817評(píng)論 0 268
  • 序言:老撾萬榮一對(duì)情侶失蹤蜡坊,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后赎败,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體秕衙,經(jīng)...
    沈念sama閱讀 44,275評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,592評(píng)論 2 327
  • 正文 我和宋清朗相戀三年僵刮,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了据忘。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,724評(píng)論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡搞糕,死狀恐怖勇吊,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情窍仰,我是刑警寧澤汉规,帶...
    沈念sama閱讀 34,409評(píng)論 4 333
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響针史,放射性物質(zhì)發(fā)生泄漏晶伦。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,052評(píng)論 3 316
  • 文/蒙蒙 一啄枕、第九天 我趴在偏房一處隱蔽的房頂上張望婚陪。 院中可真熱鬧,春花似錦频祝、人聲如沸泌参。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,815評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽沽一。三九已至,卻和暖如春窟绷,著一層夾襖步出監(jiān)牢的瞬間锯玛,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,043評(píng)論 1 266
  • 我被黑心中介騙來泰國(guó)打工兼蜈, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留攘残,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,503評(píng)論 2 361
  • 正文 我出身青樓为狸,卻偏偏與公主長(zhǎng)得像歼郭,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子辐棒,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,627評(píng)論 2 350

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