JavaScript設(shè)計(jì)模式之觀察者模式

前言

準(zhǔn)備研究一下MVVM的一些東西聘殖,由于MVVM運(yùn)用了觀察者模式的思想,因此翻開(kāi)了《JavaScript設(shè)計(jì)模式與開(kāi)發(fā)實(shí)踐》一書晕翠,將觀察者模式學(xué)習(xí)了一遍朦肘,順便有對(duì)一些常用的設(shè)計(jì)模式進(jìn)行一些了解,但還是有很多不能理解的地方嘱蛋,還需努力啊蚯姆。

一、什么是觀察者模式

觀察者模式又叫做發(fā)布—訂閱模式洒敏,是我們最常用的設(shè)計(jì)模式之一龄恋。它定義對(duì)象間的一種一對(duì)多的依賴關(guān)系,當(dāng)一個(gè)對(duì)象的狀態(tài)發(fā)生改變時(shí)凶伙,所有依賴于它的對(duì)象都將得到通知和更新郭毕。觀察者模式提供了一個(gè)訂閱模型,其中對(duì)象訂閱事件并在發(fā)生時(shí)得到通知函荣,這種模式是事件驅(qū)動(dòng)的編程基石显押,它有利益于良好的面向?qū)ο蟮脑O(shè)計(jì)。

看了上面的這段描述傻挂,可能還是不懂什么是觀察者模式乘碑。我們還可以來(lái)看看生活中的觀察者模式:現(xiàn)在房?jī)r(jià)這么高,你肯定是想要早點(diǎn)買房金拒,但你看好的樓盤還沒(méi)開(kāi)盤兽肤,因此你就將你的電話留給售樓小姐,一旦樓盤推出就讓她打電話給你绪抛。主動(dòng)權(quán)在售樓方资铡,而你只需要提供一個(gè)聯(lián)系方式就行了這樣你就不用擔(dān)心樓盤出來(lái)你不知道了,也不需要每天都打電話去問(wèn)樓盤推出了沒(méi)幢码。

二笤休、觀察者模式的使用場(chǎng)景

2.1 DOM事件

實(shí)際上,只要我們?cè)?jīng)在DOM節(jié)點(diǎn)上面綁定過(guò)事件函數(shù)蛤育,那我們就使用過(guò)觀察者模式宛官,應(yīng)為JS和DOM之間就是實(shí)現(xiàn)了一種觀察者模式葫松。

document.body.addEventListener("click", function() {
    alert("Hello World")
},false )
document.body.click() //模擬用戶點(diǎn)擊

在上面的代碼中底洗,需要監(jiān)聽(tīng)用戶點(diǎn)擊 document.body 的動(dòng)作腋么,但是我們是沒(méi)辦法預(yù)知用戶將在什么時(shí)候點(diǎn)擊的。因此我們訂閱了 document.body 的 click 事件亥揖,當(dāng) body 節(jié)點(diǎn)被點(diǎn)擊時(shí)珊擂,body 節(jié)點(diǎn)便會(huì)向訂閱者發(fā)布 "Hello World" 消息。

2.2自定義事件

除了DOM事件外费变,我們還可以實(shí)現(xiàn)一些自定義事件摧扇,這種依靠自定義時(shí)間完成的觀察者模式可以用于任何的JavaScript代碼中。

示例:

const event = {
    clientList: [],
    listen: function(key , fn) {
        if (this.clientListen[key]) {
            this.clientList[key] = []
        }
        this.clientList[key].push(fn)
    },
    trigger: function() {
        const key = Array.prototype.shift.call(arguments)
        const fns = this.clientList[key]
        if (!fns || fns.length === 0 ) {
            return false
        }
        for (let i = 0, fn ;fn = fns[i++];) {
            fn.apply(this, arguments)
        }
    },
    remove : function(key , fn) {
        const fns = this.clientList[key]
        if (!fns) {
            return false
        }
        if (!fn) {
            fns && (fns.length = 0)
        } else {
            for (let l = fns.length - 1; l>=0; l--) {
                const _fn = fns[l]
                if ( _fn ===fn) {
                    fns.splice(l, 1)
                }
            }
        }
}

const installEvent = (obj) => {
    for (let i in event) {
        obj[i] = event[i]
    }
}

然后就能增加發(fā)布和訂閱功能了:

const events = {}
installEvent(events)
// 訂閱信息
events.listen('newMessage',fn1 = (say) => {
    console.log('say:' + say)
})
// 發(fā)布信息
events.trigger('newMessage',"Hello world")
//移除訂閱
events.remove('newMessage',fn1)

三挚歧、觀察者模式的不足

觀察者模式的有點(diǎn)非常明顯:一是時(shí)間上的解耦扛稽,而是對(duì)象之間的解耦。既可用于異步編程中滑负,也可以用幫助我們完成更松耦合的代碼編寫在张。但它仍然有所不足:

  • 創(chuàng)建訂閱者本身要消耗一定的時(shí)間和內(nèi)存
  • 當(dāng)訂閱一個(gè)消息時(shí),也許此消息并沒(méi)有發(fā)生矮慕,但這個(gè)訂閱者會(huì)始終存在內(nèi)存中帮匾。
  • 觀察者模式弱化了對(duì)象之間的聯(lián)系,這本是好事情痴鳄,但如果過(guò)度使用瘟斜,對(duì)象與對(duì)象之間的聯(lián)系也會(huì)被隱藏的很深,會(huì)導(dǎo)致項(xiàng)目的難以跟蹤維護(hù)和理解痪寻。

原文鏈接JavaScript設(shè)計(jì)模式之觀察者模式

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末螺句,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子槽华,更是在濱河造成了極大的恐慌壹蔓,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,214評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件猫态,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡披摄,警方通過(guò)查閱死者的電腦和手機(jī)亲雪,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,307評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)疚膊,“玉大人义辕,你說(shuō)我怎么就攤上這事≡⒌粒” “怎么了灌砖?”我有些...
    開(kāi)封第一講書人閱讀 152,543評(píng)論 0 341
  • 文/不壞的土叔 我叫張陵璧函,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我基显,道長(zhǎng)蘸吓,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書人閱讀 55,221評(píng)論 1 279
  • 正文 為了忘掉前任撩幽,我火速辦了婚禮库继,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘窜醉。我一直安慰自己宪萄,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,224評(píng)論 5 371
  • 文/花漫 我一把揭開(kāi)白布榨惰。 她就那樣靜靜地躺著拜英,像睡著了一般。 火紅的嫁衣襯著肌膚如雪琅催。 梳的紋絲不亂的頭發(fā)上聊记,一...
    開(kāi)封第一講書人閱讀 49,007評(píng)論 1 284
  • 那天,我揣著相機(jī)與錄音恢暖,去河邊找鬼排监。 笑死,一個(gè)胖子當(dāng)著我的面吹牛杰捂,可吹牛的內(nèi)容都是我干的舆床。 我是一名探鬼主播,決...
    沈念sama閱讀 38,313評(píng)論 3 399
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼嫁佳,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼挨队!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起蒿往,我...
    開(kāi)封第一講書人閱讀 36,956評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤盛垦,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后瓤漏,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體腾夯,經(jīng)...
    沈念sama閱讀 43,441評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,925評(píng)論 2 323
  • 正文 我和宋清朗相戀三年蔬充,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了蝶俱。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,018評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡饥漫,死狀恐怖榨呆,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情庸队,我是刑警寧澤积蜻,帶...
    沈念sama閱讀 33,685評(píng)論 4 322
  • 正文 年R本政府宣布闯割,位于F島的核電站,受9級(jí)特大地震影響竿拆,放射性物質(zhì)發(fā)生泄漏宙拉。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,234評(píng)論 3 307
  • 文/蒙蒙 一如输、第九天 我趴在偏房一處隱蔽的房頂上張望鼓黔。 院中可真熱鬧,春花似錦不见、人聲如沸澳化。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 30,240評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)缎谷。三九已至,卻和暖如春灶似,著一層夾襖步出監(jiān)牢的瞬間列林,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 31,464評(píng)論 1 261
  • 我被黑心中介騙來(lái)泰國(guó)打工酪惭, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留希痴,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,467評(píng)論 2 352
  • 正文 我出身青樓春感,卻偏偏與公主長(zhǎng)得像砌创,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子鲫懒,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,762評(píng)論 2 345

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

  • 工廠模式類似于現(xiàn)實(shí)生活中的工廠可以產(chǎn)生大量相似的商品嫩实,去做同樣的事情,實(shí)現(xiàn)同樣的效果;這時(shí)候需要使用工廠模式窥岩。簡(jiǎn)單...
    舟漁行舟閱讀 7,718評(píng)論 2 17
  • ??JavaScript 與 HTML 之間的交互是通過(guò)事件實(shí)現(xiàn)的颂翼。 ??事件晃洒,就是文檔或?yàn)g覽器窗口中發(fā)生的一些特...
    霜天曉閱讀 3,473評(píng)論 1 11
  • 文 / 圓靖 炎熱的夏季剛過(guò),處暑沒(méi)幾日就迎來(lái)一周的陰雨天氣疚鲤。恰逢“七夕”锥累,原本借此機(jī)會(huì)與老徐的戶外“玫瑰之約”也...
    圓立青閱讀 359評(píng)論 3 3
  • 實(shí)現(xiàn)收入指數(shù)型增長(zhǎng)的第三個(gè)關(guān)鍵點(diǎn),我們總結(jié)為強(qiáng)行融入更好的圈子集歇。 有個(gè)朋友發(fā)微信朋友圈得瑟,說(shuō)竟然發(fā)現(xiàn)跟王石在同一...
    承謙閱讀 169評(píng)論 0 0
  • 窗外的夜色慢慢蔓延,星光開(kāi)始點(diǎn)綴這片漆黑的夜空纺荧。天空的西北角掛著一輪新月被無(wú)聊的閑云調(diào)弄著旭愧,空氣中伴隨著大雨初霽的...
    木樨Leo閱讀 534評(píng)論 13 4