摘要:在看的實現之前。實現響應式的關鍵有三個遍歷中的屬性。在方法中設置核心數據劫持每個屬性都有一個自己的消息訂閱起用于訂制該屬性上的所有觀察者觀察者,通過實現對響應屬性的監(jiān)聽觀察。觀察得到結果后,主動觸發(fā)自己的回調可以去看看的這三部分源碼。
本來vue的響應式應該才是重中之重。但是網上的文章很多很多。在看computed的實現之前。肯定還是要把vue的響應式如何實現好好看一下。或者說兩者根本就是一樣的東西。這邊推薦幾篇文章關于vue的響應式。
vue響應式簡單實現
vue慕課響應式手記
還是看看官網對于響應式的解釋:
總的來說。vue實現響應式的關鍵有三個:watcher,dep,observe;
observe:遍歷data中的屬性。在get,set方法中設置核心數據劫持
dep:每個屬性都有一個自己的dep(消息訂閱起)用于訂制該屬性上的所有觀察者
watcher:觀察者,通過dep實現對響應屬性的監(jiān)聽觀察。觀察得到結果后,主動觸發(fā)自己的回調
可以去看看vue2.3的這三部分源碼。中間還是有很多精美的設計。比如一個全局唯一的Dep.target,在任何時候都是唯一的值。以確保同一時間只有一個觀察者在訂閱。再比如,watcher中也會存下相關的訂閱器,實現去重和實現同一個觀察者的分組(這里是實現computed的關鍵),再如。watcher中的id也會唯一。用于異步更新的時候不同時出發(fā)相同的訂閱。仔細看看會收獲不小。改天我把所有的響應式的代碼也整理一下。
在理解了響應式的情況下。我們來看看computed的實現。最簡單的一個demo如下:
{{computeA}}
我們來從源碼的角度看看發(fā)生了什么:
在初始化實例創(chuàng)建響應式的時候。對options中的computed做了特殊處理:
function initComputed (vm, computed) { var watchers = vm._computedWatchers = Object.create(null); for (var key in computed) { var userDef = computed[key]; var getter = typeof userDef === "function" ? userDef : userDef.get; { if (getter === undefined) { warn( ("No getter function has been defined for computed property "" + key + ""."), vm ); getter = noop; } } // create internal watcher for the computed property. watchers[key] = new Watcher(vm, getter, noop, computedWatcherOptions);//為每一個computed項目訂制一個watcher // component-defined computed properties are already defined on the // component prototype. We only need to define computed properties defined // at instantiation here. if (!(key in vm)) { defineComputed(vm, key, userDef); } else { if (key in vm.$data) { warn(("The computed property "" + key + "" is already defined in data."), vm); } else if (vm.$options.props && key in vm.$options.props) { warn(("The computed property "" + key + "" is already defined as a prop."), vm); } } function defineComputed (target, key, userDef) { if (typeof userDef === "function") { sharedPropertyDefinition.get = createComputedGetter(key); sharedPropertyDefinition.set = noop; } else { sharedPropertyDefinition.get = userDef.get ? userDef.cache !== false ? createComputedGetter(key) : userDef.get : noop; sharedPropertyDefinition.set = userDef.set ? userDef.set : noop; } Object.defineProperty(target, key, sharedPropertyDefinition); } function createComputedGetter (key) {//構造該computed的get函數 return function computedGetter () { var watcher = this._computedWatchers && this._computedWatchers[key]; if (watcher) { if (watcher.dirty) { watcher.evaluate();//收集該watcher的訂閱 } if (Dep.target) { watcher.depend();//同一為這一組訂閱再加上組件re-render的訂閱(該訂閱負責更新組件) } return watcher.value } } }
總的來說。理解了響應式的構建之后。再來看computed的實現還是很直觀的。組件初始化的時候。computed項和data中的分別建立響應式。data中的數據直接對屬性的get,set做數據攔截。而computed則建立一個新的watcher,在組件渲染的時候。先touch一下這個computed的getter函數。將這個watcher訂閱起來。這里相當于這個computed的watcher訂閱了firstname和lastname。touch完后。Dep.target此時又變?yōu)橹澳莻€用于更新組件的。再通過watcher.depend()將這個組統(tǒng)一加上這個訂閱。這樣一旦firstname和lastname變了。同時會觸發(fā)兩個訂閱更新。其中一個便是更新組件。重新re-render的函數。感覺看的還不夠細啊
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規(guī)行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/83565.html
摘要:不多廢話,先說結論小程序組件寫法這里就不再介紹。在官方文檔中,我們可以看到使用構造器構造頁面事實上,小程序的頁面也可以視為自定義組件。經過一番測試,得出結果為為了簡便。畢竟官方標準,不用擔心其他一系列后續(xù)問題。 在開發(fā)小程序的時候,我們總是期望用以往的技術規(guī)范和語法特點來書寫當前的小程序,所以才會有各色的小程序框架,例如 mpvue、taro 等這些編譯型框架。當然這些框架本身對于新開...
摘要:了解之后我們來實現它,同樣的為了方便理解我寫成了一個類這里的一般是的實例將屬性代理到實例下的構造函數我們實現了代理屬性和更新計算屬性的值,同時依賴沒變化時,也是不會觸發(fā)的更新,解決了以上的個問題。 看這篇之前,如果沒有看過之前的文章,移步拉到文章末尾查看之前的文章。 回顧 先捋一下,之前我們實現的 Vue 類,主要有一下的功能: 屬性和方法的代理 proxy 監(jiān)聽屬性 watche...
摘要:前言源碼解析一模版渲染源碼解析二雙向綁定官網給出的如下結果源碼分析判斷參數是否包含屬性本例中本例中和是函數監(jiān)聽計算屬性設置,延遲執(zhí)行的方法設置可以通過本例方式訪問計算屬性對象初始化時會針對屬性的所有值分別一個對象,在源碼解析二中有詳細介 前言 1、Vue源碼解析(一)-模版渲染2、Vue源碼解析(二)-MVVM雙向綁定 demo 官網給出的demo如下 new Vue({ el...
閱讀 1661·2019-08-30 13:04
閱讀 2213·2019-08-30 12:59
閱讀 1775·2019-08-29 18:34
閱讀 1868·2019-08-29 17:31
閱讀 1263·2019-08-29 15:42
閱讀 3544·2019-08-29 15:37
閱讀 2864·2019-08-29 13:45
閱讀 2777·2019-08-26 13:57