摘要:典型實現(xiàn)例子售樓處的例子一步步實現(xiàn)發(fā)布訂閱模式首先指定好誰充當發(fā)布者售樓處然后給發(fā)布者添加一個緩存列表,用語存放回調(diào)函數(shù),以便通知訂閱者售樓處花名冊。最后發(fā)布消息的時候,發(fā)布者會遍歷這個緩存列表,依次觸發(fā)里面存放的訂閱者的回調(diào)函數(shù)。
概念
發(fā)布-訂閱模式又稱為觀察者模式,它定義的是一種一對多的依賴關系,當一個狀態(tài)發(fā)生改變的時候,所有以來這個狀態(tài)的對象都會得到通知。
生活中的發(fā)布-訂閱模式上面事發(fā)布-訂閱模式的一個比較正式的解釋,可能這個解釋不大好理解。所以我們通過實際生活中的例子來理解。
比如看中了一套房子,等到去了售樓處的說以后才被告知房子已經(jīng)售罄了。但是售樓小姐告知,將來會有尾盤推出。具體什么時候推出,目前沒人知道。
但是買家又不想頻繁的跑,于是就把自己的電話號碼登記在售樓處,在登記的花名冊上有很多類似的買家。售樓小姐答應買家,新的房源一出來就一一通知買家。
所以上面就是一個發(fā)布訂閱模式的簡單例子。購房者(訂閱者)訂閱房源信息,售樓處(發(fā)布者)發(fā)布新房源消息給購房者(訂閱者),購房者(訂閱者)接收到消息后作出相應的反應。
適用性發(fā)布訂閱模式可以廣泛的應用于異步編程中。
發(fā)布訂閱模式可以取代對象之間的硬編碼通知機制。
典型實現(xiàn)例子1、售樓處的例子
一步步實現(xiàn)發(fā)布訂閱模式:
首先指定好誰充當發(fā)布者(售樓處)
然后給發(fā)布者添加一個緩存列表,用語存放回調(diào)函數(shù),以便通知訂閱者(售樓處花名冊)。
最后發(fā)布消息的時候,發(fā)布者會遍歷這個緩存列表,依次觸發(fā)里面存放的訂閱者的回調(diào)函數(shù)。
let salesOffices = {} // 售樓處 salesOffices.books = [] // 緩存列表,存放訂閱者的回調(diào)函數(shù)。 // 增加訂閱者 salesOffices.listen = function(fn) { this.books.push(fn) // 訂閱的消息添加近緩存列表里面 } salesOffices.trigger = function() { // 發(fā)布消息 for (let i = 0, fn; (fn = salesOffices.books[i++]); ) { fn.apply(this, arguments) // arguments 是發(fā)布消息的時候帶上的參數(shù) } } salesOffices.listen(function(price, squareMeter) { // 購買者a console.log(`價格是:${price}`) console.log(`面積大?。?{squareMeter}`) }) salesOffices.listen(function(price, squareMeter) { // 購買者b console.log(`價格是:${price}`) console.log(`面積大?。?{squareMeter}`) }) salesOffices.trigger(2000000, 88) salesOffices.trigger(3000000, 128)
上面實現(xiàn)了一個最簡單的發(fā)布訂閱模式??隙ㄟ€有很多問題的,例如訂閱者只訂閱了某一個消息,但是上面會把所有消息發(fā)給每一個訂閱者。所以還得通過其他的方式讓訂閱者只訂閱自己感興趣的消息。
2、vue 對發(fā)布訂閱模式的使用
我們都知道 Vue 有個最顯著的特性,便是侵入性不是很強的響應式系統(tǒng)。這個特性就是對發(fā)布訂閱模式非常好的應用。我們接下來就來看看這個特性是怎么應用的。
vue 的數(shù)據(jù)初始化:
var v = new Vue({ data() { return { a: "hello" } } })
這個初始化的代碼的背后包含著發(fā)布訂閱模式的思想,接下來看看官網(wǎng)的一個圖
接下來就是網(wǎng)友的一個圖:@xuqiang521
從上圖可以看到,數(shù)據(jù)劫持的核心方法就是使用Object.defineProperty把屬性轉(zhuǎn)化成getter/setter。(因為這個是 ES5 中的方法,所以這也是 Vue 不支持 ie8 及以下瀏覽器的原因之一。)在數(shù)據(jù)傳遞變更的時候,會進入到我們封裝的Dep和Watcher中進行處理。
數(shù)據(jù)不緊緊是基本類型的數(shù)據(jù),也有可能是對象或者數(shù)組?;绢愋偷臄?shù)據(jù)和對象的處理起來比較簡單。
walk(obj) { const keys = Object.keys(obj) for (let i = 0; i < keys.length; ++i) { defineReactive(obj, keys[i], obj[keys[i]]) } }
核心的劫持相關函數(shù)以及屬性的訂閱和發(fā)布
/** * Define a reactive property on an Object. */ export function defineReactive( obj: Object, key: string, val: any, customSetter?: Function ) { /*在閉包中定義一個dep對象*/ const dep = new Dep() const property = Object.getOwnPropertyDescriptor(obj, key) if (property && property.configurable === false) { return } /*如果之前該對象已經(jīng)預設了getter以及setter函數(shù)則將其取出來,新定義的getter/setter中會將其執(zhí)行,保證不會覆蓋之前已經(jīng)定義的getter/setter。*/ // cater for pre-defined getter/setters const getter = property && property.get const setter = property && property.set /*對象的子對象遞歸進行observe并返回子節(jié)點的Observer對象*/ let childOb = observe(val) Object.defineProperty(obj, key, { enumerable: true, configurable: true, get: function reactiveGetter() { /*如果原本對象擁有getter方法則執(zhí)行*/ const value = getter ? getter.call(obj) : val if (Dep.target) { /*進行依賴收集*/ dep.depend() if (childOb) { /*子對象進行依賴收集,其實就是將同一個watcher觀察者實例放進了兩個depend中,一個是正在本身閉包中的depend,另一個是子元素的depend*/ childOb.dep.depend() } if (Array.isArray(value)) { /*是數(shù)組則需要對每一個成員都進行依賴收集,如果數(shù)組的成員還是數(shù)組,則遞歸。*/ dependArray(value) } } return value }, set: function reactiveSetter(newVal) { /*通過getter方法獲取當前值,與新值進行比較,一致則不需要執(zhí)行下面的操作*/ const value = getter ? getter.call(obj) : val /* eslint-disable no-self-compare */ if (newVal === value || (newVal !== newVal && value !== value)) { return } /* eslint-enable no-self-compare */ if (process.env.NODE_ENV !== "production" && customSetter) { customSetter() } if (setter) { /*如果原本對象擁有setter方法則執(zhí)行setter*/ setter.call(obj, newVal) } else { val = newVal } /*新的值需要重新進行observe,保證數(shù)據(jù)響應式*/ childOb = observe(newVal) /*dep對象通知所有的觀察者*/ dep.notify() } }) }
最開始在初始化的時候是對 data 里面的數(shù)據(jù)就開始劫持監(jiān)聽了。初始化的時候就調(diào)用了observe方法
/** * Attempt to create an observer instance for a value, * returns the new observer if successfully observed, * or the existing observer if the value already has one. */ /* 嘗試創(chuàng)建一個Observer實例(__ob__),如果成功創(chuàng)建Observer實例則返回新的Observer實例,如果已有Observer實例則返回現(xiàn)有的Observer實例。 */ export function observe(value: any, asRootData: ?boolean): Observer | void { /*判斷是否是一個對象*/ if (!isObject(value)) { return } let ob: Observer | void /*這里用__ob__這個屬性來判斷是否已經(jīng)有Observer實例,如果沒有Observer實例則會新建一個Observer實例并賦值給__ob__這個屬性,如果已有Observer實例則直接返回該Observer實例*/ if (hasOwn(value, "__ob__") && value.__ob__ instanceof Observer) { ob = value.__ob__ } else if ( /*這里的判斷是為了確保value是單純的對象,而不是函數(shù)或者是Regexp等情況。*/ observerState.shouldConvert && !isServerRendering() && (Array.isArray(value) || isPlainObject(value)) && Object.isExtensible(value) && !value._isVue ) { ob = new Observer(value) } if (asRootData && ob) { /*如果是根數(shù)據(jù)則計數(shù),后面Observer中的observe的asRootData非true*/ ob.vmCount++ } return ob }
上面的數(shù)據(jù)observe之后返回的就是一個 Observer 的實例
ob = new Observer(value) return ob
在第一步數(shù)據(jù)劫持的時候,數(shù)據(jù)的獲取或者修改的時候,都會做出對應的操作。這些操作的目的很簡單,就是“通知”到“中轉(zhuǎn)站”。這個“中轉(zhuǎn)站”主要就是對數(shù)據(jù)的變更起通知作用以及存放依賴這些數(shù)據(jù)的“地方”。
這個"中轉(zhuǎn)站"就是由"Dep"和“Watcher” 類構成的。每個被劫持的數(shù)據(jù)都會產(chǎn)生一個這樣的“中轉(zhuǎn)站”
Dep,全名 Dependency,從名字我們也能大概看出 Dep 類是用來做依賴收集的,但是也有通知對應的訂閱者的作用 ,讓它執(zhí)行自己的操作,具體怎么收集呢?
/** * A dep is an observable that can have multiple * directives subscribing to it. */ export default class Dep { static target: ?Watcher id: number subs: Arrayconstructor() { this.id = uid++ this.subs = [] } /*添加一個觀察者對象*/ addSub(sub: Watcher) { this.subs.push(sub) } /*移除一個觀察者對象*/ removeSub(sub: Watcher) { remove(this.subs, sub) } /*依賴收集,當存在Dep.target的時候添加觀察者對象*/ depend() { if (Dep.target) { Dep.target.addDep(this) } } /*通知所有訂閱者*/ notify() { // stabilize the subscriber list first const subs = this.subs.slice() for (let i = 0, l = subs.length; i < l; i++) { subs[i].update() } } } // the current target watcher being evaluated. // this is globally unique because there could be only one // watcher being evaluated at any time. Dep.target = null /*依賴收集完需要將Dep.target設為null,防止后面重復添加依賴。*/ const targetStack = [] export function pushTarget(_target: Watcher) { if (Dep.target) targetStack.push(Dep.target) // 改變目標指向 Dep.target = _target } export function popTarget() { // 刪除當前目標,重算指向 Dep.target = targetStack.pop() }
代碼很簡短,但它做的事情卻很重要
定義 subs 數(shù)組,用來收集訂閱者 Watcher
當劫持到數(shù)據(jù)變更的時候,通知訂閱者 Watcher 進行 update 操作
Watcher 就是訂閱者(觀察者)。 主要的作用就是就是訂閱 Dep(每個屬性都會有一個 dep),當 Dep 發(fā)出消息傳遞(notify)的時候,所以訂閱著 Dep 的 Watchers 會進行自己的 update 操作。
export default class Watcher { vm: Component expression: string cb: Function id: number deep: boolean user: boolean lazy: boolean sync: boolean dirty: boolean active: boolean deps: ArraynewDeps: Array depIds: ISet newDepIds: ISet getter: Function value: any constructor( vm: Component, expOrFn: string | Function, cb: Function, options?: Object ) { this.vm = vm /*_watchers存放訂閱者實例*/ vm._watchers.push(this) // options if (options) { this.deep = !!options.deep this.user = !!options.user this.lazy = !!options.lazy this.sync = !!options.sync } else { this.deep = this.user = this.lazy = this.sync = false } this.cb = cb this.id = ++uid // uid for batching this.active = true this.dirty = this.lazy // for lazy watchers this.deps = [] this.newDeps = [] this.depIds = new Set() this.newDepIds = new Set() this.expression = process.env.NODE_ENV !== "production" ? expOrFn.toString() : "" // parse expression for getter /*把表達式expOrFn解析成getter*/ if (typeof expOrFn === "function") { this.getter = expOrFn } else { this.getter = parsePath(expOrFn) if (!this.getter) { this.getter = function() {} process.env.NODE_ENV !== "production" && warn( `Failed watching path: "${expOrFn}" ` + "Watcher only accepts simple dot-delimited paths. " + "For full control, use a function instead.", vm ) } } this.value = this.lazy ? undefined : this.get() } /** * Evaluate the getter, and re-collect dependencies. */ /*獲得getter的值并且重新進行依賴收集*/ get() { /*將自身watcher觀察者實例設置給Dep.target,用以依賴收集。*/ pushTarget(this) let value const vm = this.vm /* 執(zhí)行了getter操作,看似執(zhí)行了渲染操作,其實是執(zhí)行了依賴收集。 在將Dep.target設置為自生觀察者實例以后,執(zhí)行getter操作。 譬如說現(xiàn)在的的data中可能有a、b、c三個數(shù)據(jù),getter渲染需要依賴a跟c, 那么在執(zhí)行getter的時候就會觸發(fā)a跟c兩個數(shù)據(jù)的getter函數(shù), 在getter函數(shù)中即可判斷Dep.target是否存在然后完成依賴收集, 將該觀察者對象放入閉包中的Dep的subs中去。 */ if (this.user) { try { value = this.getter.call(vm, vm) } catch (e) { handleError(e, vm, `getter for watcher "${this.expression}"`) } } else { value = this.getter.call(vm, vm) } // "touch" every property so they are all tracked as // dependencies for deep watching /*如果存在deep,則觸發(fā)每個深層對象的依賴,追蹤其變化*/ if (this.deep) { /*遞歸每一個對象或者數(shù)組,觸發(fā)它們的getter,使得對象或數(shù)組的每一個成員都被依賴收集,形成一個“深(deep)”依賴關系*/ traverse(value) } /*將觀察者實例從target棧中取出并設置給Dep.target*/ popTarget() this.cleanupDeps() return value } /** * Add a dependency to this directive. */ /*添加一個依賴關系到Deps集合中*/ addDep(dep: Dep) { const id = dep.id if (!this.newDepIds.has(id)) { this.newDepIds.add(id) this.newDeps.push(dep) if (!this.depIds.has(id)) { dep.addSub(this) } } } /** * Clean up for dependency collection. */ /*清理依賴收集*/ cleanupDeps() { /*移除所有觀察者對象*/ let i = this.deps.length while (i--) { const dep = this.deps[i] if (!this.newDepIds.has(dep.id)) { dep.removeSub(this) } } let tmp = this.depIds this.depIds = this.newDepIds this.newDepIds = tmp this.newDepIds.clear() tmp = this.deps this.deps = this.newDeps this.newDeps = tmp this.newDeps.length = 0 } /** * Subscriber interface. * Will be called when a dependency changes. */ /* 調(diào)度者接口,當依賴發(fā)生改變的時候進行回調(diào)。 */ update() { /* istanbul ignore else */ if (this.lazy) { this.dirty = true } else if (this.sync) { /*同步則執(zhí)行run直接渲染視圖*/ this.run() } else { /*異步推送到觀察者隊列中,由調(diào)度者調(diào)用。*/ queueWatcher(this) } } /** * Scheduler job interface. * Will be called by the scheduler. */ /* 調(diào)度者工作接口,將被調(diào)度者回調(diào)。 */ run() { if (this.active) { const value = this.get() if ( value !== this.value || // Deep watchers and watchers on Object/Arrays should fire even // when the value is the same, because the value may // have mutated. /* 即便值相同,擁有Deep屬性的觀察者以及在對象/數(shù)組上的觀察者應該被觸發(fā)更新,因為它們的值可能發(fā)生改變。 */ isObject(value) || this.deep ) { // set new value const oldValue = this.value /*設置新的值*/ this.value = value /*觸發(fā)回調(diào)渲染視圖*/ if (this.user) { try { this.cb.call(this.vm, value, oldValue) } catch (e) { handleError(e, this.vm, `callback for watcher "${this.expression}"`) } } else { this.cb.call(this.vm, value, oldValue) } } } } /** * Evaluate the value of the watcher. * This only gets called for lazy watchers. */ /*獲取觀察者的值*/ evaluate() { this.value = this.get() this.dirty = false } /** * Depend on all deps collected by this watcher. */ /*收集該watcher的所有deps依賴*/ depend() { let i = this.deps.length while (i--) { this.deps[i].depend() } } /** * Remove self from all dependencies" subscriber list. */ /*將自身從所有依賴收集訂閱列表刪除*/ teardown() { if (this.active) { // remove self from vm"s watcher list // this is a somewhat expensive operation so we skip it // if the vm is being destroyed. /*從vm實例的觀察者列表中將自身移除,由于該操作比較耗費資源,所以如果vm實例正在被銷毀則跳過該步驟。*/ if (!this.vm._isBeingDestroyed) { remove(this.vm._watchers, this) } let i = this.deps.length while (i--) { this.deps[i].removeSub(this) } this.active = false } } }
通過上面對 vue 的響應系統(tǒng)的 學習,就可以了解到這個發(fā)布訂閱模式就是這樣的:
Dep 負責收集所有相關的的訂閱者 Watcher ,具體誰不用管,具體有多少也不用管,只需要根據(jù) target 指向的計算去收集訂閱其消息的 Watcher 即可,然后做好消息發(fā)布 notify 即可。
Watcher 負責訂閱 Dep ,并在訂閱的時候讓 Dep 進行收集,接收到 Dep 發(fā)布的消息時,做好其 update 操作即可。
3、vue 中更多的應用
vue 中還有個組件之間的時間傳遞也是用到了發(fā)布訂閱模式。
$emit 負責發(fā)布消息, $on 負責消費消息(執(zhí)行 cbs 里面的事件)
Vue.prototype.$on = function( event: string | Array總結, fn: Function ): Component { const vm: Component = this if (Array.isArray(event)) { for (let i = 0, l = event.length; i < l; i++) { this.$on(event[i], fn) } } else { ;(vm._events[event] || (vm._events[event] = [])).push(fn) } return vm } Vue.prototype.$emit = function(event: string): Component { const vm: Component = this let cbs = vm._events[event] if (cbs) { cbs = cbs.length > 1 ? toArray(cbs) : cbs const args = toArray(arguments, 1) for (let i = 0, l = cbs.length; i < l; i++) { cbs[i].apply(vm, args) } } return vm }
本文通過對 vue 相關源碼的學習,了解了發(fā)布訂閱模式(觀察者模式)的概念和應用。還了解了該模式的 一些優(yōu)缺點:
時間上的解耦,對象之間的解耦。
創(chuàng)建訂閱者本身會消耗一定的時間和內(nèi)存,并且訂閱者訂閱一個消息后,該消息一直不發(fā)生的話,那么該訂閱者 會一直存在在內(nèi)存中
感謝從源碼角度再看數(shù)據(jù)綁定
《javascript 設計模式與開發(fā)實踐》
文章版權歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/107602.html
摘要:下面我們會向大家解釋清楚為什么這個這么重要,以及它和的響應式數(shù)據(jù)流有什么關系。源碼前面鋪墊這么多就是希望大家能理解接下來要講的響應式數(shù)據(jù)流。總結講到這里大家應該都能夠明白的響應式數(shù)據(jù)流是如何實現(xiàn)的。 Vue、React介紹 目前前端社區(qū)比較推崇的框架有Vue 和 React,公司內(nèi)部許多端都自發(fā)的將原有的老技術方案(widget + jQuery)遷移到 Vue / React上了。我...
摘要:今天,就我們就來一步步解析響應式的原理,并且來實現(xiàn)一個簡單的。當然,這個也只是一個簡單的,來說明響應式的原理,真實的源碼會更加復雜,因為加了很多其他邏輯。接下來我可能會將其與聯(lián)系起來,實現(xiàn)和語法。 從很久之前就已經(jīng)接觸過了angularjs了,當時就已經(jīng)了解到,angularjs是通過臟檢查來實現(xiàn)數(shù)據(jù)監(jiān)測以及頁面更新渲染。之后,再接觸了vue.js,當時也一度很好奇vue.js是如何監(jiān)...
摘要:或許以前認為訂閱發(fā)布模式是觀察者模式的一種別稱,但是發(fā)展至今,概念已經(jīng)有了不少區(qū)別。參考文章訂閱發(fā)布模式和觀察者模式真的不一樣 首選我們需要先了解兩者的定義和實現(xiàn)的方式,才能更好的區(qū)分兩者的不同點。 或許以前認為訂閱發(fā)布模式是觀察者模式的一種別稱,但是發(fā)展至今,概念已經(jīng)有了不少區(qū)別。 訂閱發(fā)布模式 在軟件架構中,發(fā)布-訂閱是一種消息范式,消息的發(fā)送者(稱為發(fā)布者)不會將消息直接發(fā)送給特...
閱讀 2694·2021-10-22 09:55
閱讀 2026·2021-09-27 13:35
閱讀 1280·2021-08-24 10:02
閱讀 1507·2019-08-30 15:55
閱讀 1209·2019-08-30 14:13
閱讀 3484·2019-08-30 13:57
閱讀 1983·2019-08-30 11:07
閱讀 2462·2019-08-29 17:12