Vue之Dep和Observer的用法及說明
Vue 中響應式系統(tǒng)利用了訂閱發(fā)布模式來實現基本的邏輯。本文將介紹其中的兩個重要角色,他們就是Dep和Observer。其中Observer 是觀察者和 Dep是訂閱收集和發(fā)布者。另外watcher是作為訂閱者的角色。
本文將重點將Observer和Dep。
一:Observer
vue 通過Observer 構造函數,為響應式變量添加訪問和賦值的get set的回調。
? var Observer = function Observer (value) { ? ? this.value = value; ? ? this.dep = new Dep(); ? ? this.vmCount = 0; ? ? def(value, '__ob__', this); ? ? if (Array.isArray(value)) { ? ? ? if (hasProto) { ? ? ? ? protoAugment(value, arrayMethods); ? ? ? } else { ? ? ? ? copyAugment(value, arrayMethods, arrayKeys); ? ? ? } ? ? ? this.observeArray(value); ? ? } else { ? ? ? this.walk(value); // 這是非數組值添加get set 回調 ? ? } ? };
批量為obj上的key添加get set的回調
? Observer.prototype.walk = function walk (obj) { ? ? var keys = Object.keys(obj); ? ? for (var i = 0; i < keys.length; i++) { ? ? ? defineReactive$$1(obj, keys[i]); ? ? } ? };
// 這個是為obj上的key添加get set方法的核心邏輯 ? function defineReactive$$1 ( ? ? obj, ? ? key, ? ? val, ? ? customSetter, ? ? shallow ? ) { ? ? var dep = new Dep(); ? ? var property = Object.getOwnPropertyDescriptor(obj, key); ? ? if (property && property.configurable === false) { ? ? ? return ? ? } ? ? // cater for pre-defined getter/setters ? ? var getter = property && property.get; ? ? var setter = property && property.set; ? ? if ((!getter || setter) && arguments.length === 2) { ? ? ? val = obj[key]; ? ? } ? ? var childOb = !shallow && observe(val); ? ? Object.defineProperty(obj, key, { ? ? ? enumerable: true, ? ? ? configurable: true, ? ? ? get: function reactiveGetter () { ? ? ? ? ?// 省略N行 ? ? ? }, ? ? ? set: function reactiveSetter (newVal) { ? ? ? ?// 省略N行 ? ? ? } ? ? }); ? }
有了這一層,當數據獲取和修改就會觸發(fā)這里的get 和 set
二:Dep 依賴關系
變量能夠響應數據的獲取和修改對于一個透明處理dom更新的框架來說是不夠的,框架還需要處理變量和更新dom的watcher的依賴關系。Dep就是為了干這個事情的。
在之前的文章里,介紹過Watcher,vue中的組件在掛載前,都會基于組件的render函數,生成一個Watcher實例,并執(zhí)行render函數來進行渲染
渲染dom的時候我們需要讀取變量,這個時候就會觸發(fā)我們上一步Observer為每個變量里量身定做的get方法。那vue在get里做了什么事情呢?
請看下面一段代碼中的中文注釋:
function defineReactive$$1 ( ? ? obj, ? ? key, ? ? val, ? ? customSetter, ? ? shallow ? ) { ? ? var dep = new Dep(); // 創(chuàng)建一個Dep實例 ? ? var property = Object.getOwnPropertyDescriptor(obj, key); ? ? if (property && property.configurable === false) { ? ? ? return ? ? } ? ? // cater for pre-defined getter/setters ? ? var getter = property && property.get; ? ? var setter = property && property.set; ? ? if ((!getter || setter) && arguments.length === 2) { ? ? ? val = obj[key]; ? ? } ? ? var childOb = !shallow && observe(val); ? ? Object.defineProperty(obj, key, { ? ? ? enumerable: true, ? ? ? configurable: true, ? ? ? get: function reactiveGetter () { ? ? ? ? var value = getter ? getter.call(obj) : val; ? ? ? ? if (Dep.target) { ? ? ? ? ? //在此處將當前watcher加入到dep中 ? ? ? ? ? dep.depend();? ? ? ? ? ? if (childOb) { ? ? ? ? ? ? childOb.dep.depend(); ? ? ? ? ? ? if (Array.isArray(value)) { ? ? ? ? ? ? ? dependArray(value); ? ? ? ? ? ? } ? ? ? ? ? } ? ? ? ? } ? ? ? ? return value ? ? ? },
此處有一個Dep.target 存在的條件判斷,什么是Dep.target
Dep.target = null;
當watcher實例創(chuàng)建完,默認回去調用get方法來渲染組件界面
這個時候會通過pushTarget(this) 將當前watcher設置到Dep.target上去。
? Watcher.prototype.get = function get () { ? ?// 設置當前watcher到全局變量上去 ? ? pushTarget(this); ? ? var value; ? ? // 省略N行代碼 ? ? return value ? };
pushTarget的具體實現
?function pushTarget (target) { ? ? targetStack.push(target); ? ? Dep.target = target; ? }
這一步完成后,在調用render函數觸發(fā)dom中響應式變量的get的時候
下面的代碼就能夠執(zhí)行了
?if (Dep.target) { ? ? ? ? ? //在此處將當前watcher加入到dep中 ? ? ? ? ? dep.depend();?
dep.depend 干了啥呢?
接著看:
?Dep.prototype.depend = function depend () { ? ? // Dep.target 對應的就是上面的全局變量里的watcher ? ? if (Dep.target) { ? ? ? Dep.target.addDep(this); ? ? } ? };
也就是調用了watcher.addDep(this)
? Watcher.prototype.addDep = function addDep (dep) { ? ? var id = dep.id; ? ? if (!this.newDepIds.has(id)) { ? ? ? this.newDepIds.add(id); ? ? ? // ?會在watcher的實際例子里維護一個所以訂閱的dep的數組 ? ? ? this.newDeps.push(dep);? ? ? ? if (!this.depIds.has(id)) { ? ? ? ? dep.addSub(this);? ? ? ? } ? ? } ? }; ? Dep.prototype.addSub = function addSub (sub) { ? // 繞了一大圈,最終就是把watcher 塞到dep實例的subs數組里了。 ? ? this.subs.push(sub); ? };
到此,dom的渲染訂閱者和數據的觀察發(fā)布者就關聯(lián)上了。
等用戶修改數據的時候,觸發(fā)為響應數據設計的set 回調函數
set: function reactiveSetter (newVal) { ? ? ? ? var 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 (customSetter) { ? ? ? ? ? customSetter(); ? ? ? ? } ? ? ? ? // #7981: for accessor properties without setter ? ? ? ? if (getter && !setter) { return } ? ? ? ? if (setter) { ? ? ? ? ? setter.call(obj, newVal); ? ? ? ? } else { ? ? ? ? ? val = newVal; ? ? ? ? } ? ? ? ? childOb = !shallow && observe(newVal); ? ? ? ? // 重點看這里,這里是發(fā)布者發(fā)布數據變化的消息 ? ? ? ? // dep通知所有依賴的watcher ? ? ? ? dep.notify(); ? ? ? } ? ? });
dep.notify做了啥事呢?
? Dep.prototype.notify = function notify () { ? ? // stabilize the subscriber list first ? ? // subs就是上面get回調里,我們用來加入watcher依賴的數組 ? ? var subs = this.subs.slice();? ? ? if (!config.async) { ? ? ? // subs aren't sorted in scheduler if not running async ? ? ? // we need to sort them now to make sure they fire in correct ? ? ? // order ? ? ? subs.sort(function (a, b) { return a.id - b.id; }); ? ? } ? ? for (var i = 0, l = subs.length; i < l; i++) { ? ? // ?遍歷調用watcher的update方法 ? ? ? subs[i].update();? ? ? } ? };
Dep 的構造函數和其他實例方法:
? // Dep的構造函數 ? var Dep = function Dep () { ? ? this.id = uid++; ? ? this.subs = []; ? }; // ?取消訂閱 ? Dep.prototype.removeSub = function removeSub (sub) { ? ? remove(this.subs, sub); ? };
其他相關
// 移除deps數組里的不在newDeps里的老的依賴,將之前newDeps收集的dep依賴丟給this.deps ? Watcher.prototype.cleanupDeps = function cleanupDeps () { ? ? var i = this.deps.length; ? ? while (i--) { ? ? ? var dep = this.deps[i]; ? ? ? if (!this.newDepIds.has(dep.id)) { ? ? ? ? dep.removeSub(this); ? ? ? } ? ? } ? ? var 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; ? };
總結
以上為個人經驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
Element-ui的table中使用fixed后出現行混亂情況的解決
這篇文章主要介紹了Element-ui的table中使用fixed后出現行混亂情況的解決方案,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-10-10