解讀Vue之怎樣把資料包裝成reactive從而達到MDV效果
這篇文章就來講講 vue 是如何把資料包裝成 reactive,從而實現 MDV(Model-Driven-View) 的效果,希望對大家有幫助。
先說明什麼叫做 reactive,簡單來說,就是將資料包裝成一種可觀測的類型,當資料產生變更的時候,我們能夠感知到。
而Vue 的相關實作程式碼全部都在core/observer
目錄下,而要自行閱讀的話,建議從core/instance/index.js
中開始。
在開始講 reactive 的具體實作之前,先說幾個物件:Watcher、Dep、Observer。
Watcher
Watcher 是 vue 實作的一個用於觀測資料的對象,具體實作在 core/observer/watcher.js
中。
這個類別主要是用來觀察方法/表達式
中引用到的資料(資料需要是 reative 的,即 data 或 props)變更,當變更後做出對應處理。先來看看 Watcher 這個類別的入參:
vm: Component,expOrFn: string | Function,cb: Function,options?: Object
解釋一下這幾個入參是幹嘛的:
- ##vm:目前這個 watcher 所屬的 VueComponent。
- expOrFn:需要監聽的 方法/表達式。舉個例子:VueComponent 的render function,或是computed 的getter 方法,再或是
- abc.bbc.aac
這種類型的字串(由於vue 的parsePath 方法是用split('.')來做的屬性分割,所以不支援
abc['bbc'])。 expOrFn 如果是方法,則直接賦值給 watcher 的 getter 屬性,如果是表達式,則會轉換成方法再給 getter。
cb:當 getter 中引用到的 data 改變的時候,就會觸發該回呼。 - options:額外參數,可以傳入的參數為包含
- deep
、
user,
lazy,
sync,這些值預設都是為false。
- deep 如果為 true,會對 getter 傳回的物件再做一次深度遍歷,進行進一步的依賴收集。
- user 是用來標記這個監聽是否由使用者透過 $watch 呼叫的。
- lazy 用於標記watcher 是否為懶執行,該屬性是給computed data 用的,當data 中的值更改的時候,不會立即計算getter 獲取新的數值,而是給該watcher 標記為
- dirty
,當該computed data 被引用的時候才會執行從而傳回新的computed data,從而減少計算量。
sync 是表示當 data 中的值改變的時候,watcher 是否同步更新數據,如果是 true,就會立即更新數值,否則在 nextTick 中更新。
core/observer/dep.js 中,程式碼量相當少,很容易理解。
defineReactive 方法,精簡的 defineReactive 方法如下。
function defineReactive(obj, key, value) { const dep = new Dep(); Object.defineProperty(obj, key, { get() { if (Dep.target) { dep.depend(); } return value } set(newValue) { value = newValue; dep.notify(); } })}
defineReactive 方法。
initData
initData 方法在 core/instance/state.js 中,而这个方法里大部分都是做一些判断,比如防止 data 里有跟 methods 里重复的命名之类的。核心其实就一行代码:
observe(data, true)
而这个 observe 方法干的事就是创建一个 Observer 对象,而 Observer 对象就像我上面说的,对 data 进行遍历,并且调用 defineReactive 方法。
就会使用 data 节点创建一个 Observer 对象,然后对 data 下的所有数据,依次进行 reactive 的处理,也就是调用 defineReactive
方法。当执行完 defineReactive 方法之后,data 里的每一个属性,都被注入了 getter 以及 setter 逻辑,并且创建了 dep 对象。至此 initData 执行完毕。
initComputed
然后是 initComputed 方法。这个方法就是处理 vue 中 computed 节点下的数据,遍历 computed 节点,获取 key 和 value,创建 watcher 对象,如果 value 是方法,实例化 watcher 的入参 expOrFn 则为 value,否则是 value.get。
function initComputed (vm: Component, computed: Object) { ... const watchers = vm._computedWatchers = Object.create(null) for (const key in computed) { const userDef = computed[key] let getter = typeof userDef === 'function' ? userDef : userDef.get ... watchers[key] = new Watcher(vm, getter, noop, { lazy: true }) if (!(key in vm)) { defineComputed(vm, key, userDef) } else if (process.env.NODE_ENV !== 'production') { ... } }}
我们知道 expOrFn 是可以为方法,也可以是字符串的。因此,通过上面的代码我们发现了一种官方文档里没有说明的用法,比如我的 data 结构如下
{ obj: { list: [{value: '123'}] } }
如果我们要在 template 中需要使用 list 中第一个节点的 value 属性 值,就写个 computed:
computed: { value: { get: 'obj.list.0.value' }}
然后在 template 中使用的时候,直接用{<!-- -->{ value }}
,这样的话,就算 list 为空,也能保证不会报错,类似于 lodash.get 的用法。OK,扯远了,回到正题上。
创建完 watcher,就通过 Object.defineProperty 把 computed 的 key 挂载到 vm 上。并且在 getter 中添加以下逻辑
if (watcher.dirty) { watcher.evaluate() } if (Dep.target) { watcher.depend() } return watcher.value
前面我有说过,computed data 的 watcher 是 lazy 的,当 computed data 中引用的 data 发生改变后,是不会立马重新计算值的,而只是标记一下 dirty 为 true,然后当这个 computed data 被引用的时候,上面的 getter 逻辑就会判断 watcher 是否为 dirty,如果是,就重新计算值。
而后面那一段watcher.depend
。则是为了收集 computed data 中用到的 data 的依赖,从而能够实现当 computed data 中引用的 data 发生更改时,也能触发到 render function 的重新执行。
depend () { let i = this.deps.length while (i--) { this.deps[i].depend() } }
mountComponent
把 data 以及 computed 都初始化好之后,则创建一个 render function 的 watcher。逻辑如下:
export function mountComponent ( vm: Component, el: ?Element, hydrating?: boolean): Component { vm.$el = el ... callHook(vm, 'beforeMount') let updateComponent ... updateComponent = () => { vm._update(vm._render(), hydrating) } ... vm._watcher = new Watcher(vm, updateComponent, noop) if (vm.$vnode == null) { vm._isMounted = true callHook(vm, 'mounted') } return vm}
可以看到,创建 watcher 时候的入参 expOrFn 为 updateComponent 方法,而 updateComponent 方法中则是执行了 render function。而这个 watcher 不是 lazy 的,因此创建该 watcher 的时候,就会立马执行 render function 了,当执行 render function 的时候。如果 template 中有使用 data,则会触发 data 的 getter 逻辑,然后执行 dep.depend() 进行依赖收集,如果 template 中有使用 computed 的参数,也会触发 computed 的 getter 逻辑,从而再收集 computed 的方法中引用的 data 的依赖。最终完成全部依赖的收集。
最后举个例子:
<template> <p>{{ test }}</p></template><script> export default { data() { return { name: 'cool' } }, computed: { test() { return this.name + 'test'; } } }</script>
初始化流程:
- 将 name 处理为 reactive,创建 dep 实例
- 将 test 绑到 vm,创建 test 的 watcher 实例 watch1,添加 getter 逻辑。
- 创建 render function 的 watcher 实例 watcher2,并且立即执行 render function。
- 执行 render function 的时候,触发到 test 的 getter 逻辑,watcher1 及 watcher2 均与 dep 创建映射关系。
name 的值变更后的更新流程:
- 遍历绑定的 watcher 列表,执行 watcher.update()。
- watcher1.dirty 置为为 true。
- watcher2 重新执行 render function,触发到 test 的 getter,因为 watcher1.dirty 为 true,因此重新计算 test 的值,test 的值更新。
- 重渲染 view
【相关推荐:《vue.js教程》】
以上是解讀Vue之怎樣把資料包裝成reactive從而達到MDV效果的詳細內容。更多資訊請關注PHP中文網其他相關文章!

熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

記事本++7.3.1
好用且免費的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強大的PHP整合開發環境

Dreamweaver CS6
視覺化網頁開發工具

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

在 Vue.js 中使用 Bootstrap 分為五個步驟:安裝 Bootstrap。在 main.js 中導入 Bootstrap。直接在模板中使用 Bootstrap 組件。可選:自定義樣式。可選:使用插件。

可以通過以下步驟為 Vue 按鈕添加函數:將 HTML 模板中的按鈕綁定到一個方法。在 Vue 實例中定義該方法並編寫函數邏輯。

Vue.js 中的 watch 選項允許開發者監聽特定數據的變化。當數據發生變化時,watch 會觸發一個回調函數,用於執行更新視圖或其他任務。其配置選項包括 immediate,用於指定是否立即執行回調,以及 deep,用於指定是否遞歸監聽對像或數組的更改。

Vue 多頁面開發是一種使用 Vue.js 框架構建應用程序的方法,其中應用程序被劃分為獨立的頁面:代碼維護性:將應用程序拆分為多個頁面可以使代碼更易於管理和維護。模塊化:每個頁面都可以作為獨立的模塊,便於重用和替換。路由簡單:頁面之間的導航可以通過簡單的路由配置來管理。 SEO 優化:每個頁面都有自己的 URL,這有助於搜索引擎優化。

Vue.js 返回上一頁有四種方法:$router.go(-1)$router.back()使用 <router-link to="/"> 組件window.history.back(),方法選擇取決於場景。

Vue.js 遍歷數組和對像有三種常見方法:v-for 指令用於遍歷每個元素並渲染模板;v-bind 指令可與 v-for 一起使用,為每個元素動態設置屬性值;.map 方法可將數組元素轉換為新數組。

NetflixusesAcustomFrameworkcalled“ Gibbon” BuiltonReact,notReactorVuedIrectly.1)TeamSperience:selectBasedonFamiliarity.2)ProjectComplexity:vueforsimplerprojects:reactforforforproproject,reactforforforcompleplexones.3)cocatizationneedneeds:reactoffipicatizationneedneedneedneedneedneeds:reactoffersizationneedneedneedneedneeds:reactoffersizatization needefersmoreflexibleise.4)

在 Vue.js 中引用 JS 文件的方法有三種:直接使用 <script> 標籤指定路徑;利用 mounted() 生命週期鉤子動態導入;通過 Vuex 狀態管理庫進行導入。
