目錄
Watcher
initData
initComputed
mountComponent
初始化流程:
name 的值变更后的更新流程:
首頁 web前端 Vue.js 解讀Vue之怎樣把資料包裝成reactive從而達到MDV效果

解讀Vue之怎樣把資料包裝成reactive從而達到MDV效果

Dec 24, 2021 pm 05:55 PM
vue

這篇文章就來講講 vue 是如何把資料包裝成 reactive,從而實現 MDV(Model-Driven-View) 的效果,希望對大家有幫助。

解讀Vue之怎樣把資料包裝成reactive從而達到MDV效果先說明什麼叫做 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:額外參數,可以傳入的參數為包含
  • deepuserlazysync,這些值預設都是為false。
      deep 如果為 true,會對 getter 傳回的物件再做一次深度遍歷,進行進一步的依賴收集。
    • user 是用來標記這個監聽是否由使用者透過 $watch 呼叫的。
    • lazy 用於標記watcher 是否為懶執行,該屬性是給computed data 用的,當data 中的值更改的時候,不會立即計算getter 獲取新的數值,而是給該watcher 標記為
    • dirty,當該computed data 被引用的時候才會執行從而傳回新的computed data,從而減少計算量。
    • sync 是表示當 data 中的值改變的時候,watcher 是否同步更新數據,如果是 true,就會立即更新數值,否則在 nextTick 中更新。
了解了入參是用來幹嘛的之後,也基本上知道 Watcher 這個物件做了啥。

Dep

Dep 則是vue 實作的一個處理依賴關係的對象,具體實作在

core/observer/dep.js 中,程式碼量相當少,很容易理解。

Dep 主要起到一個紐帶的作用,就是連接 reactive data 與 watcher,每一個 reactive data 的創建,都會隨著創建一個 dep 實例。請參閱 observer/index.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();
        }
    })}
登入後複製
建立完 dep 實例後,就會為該 data 注入 getter 和 setter 的邏輯,當該 data 被引用的時候,就會觸發 getter,而什麼時候 data 會被引用呢?就是在 watcher 執行 getter 的時候,而當 watcher 執行 getter 的時候,watcher 會被塞入 Dep.target,然後透過呼叫 dep.depend() 方法,這個資料的 dep 就和 watcher 創造了連結。

建立連線之後,當 data 被更改,觸發了 setter 邏輯。然後就可以透過 dep.notify() 通知到所有與 dep 建立了關聯的 watcher。從而讓各個 watcher 做出回應。

例如我 watch 了一個 data ,並且在一個 computed data 中引用了同一個 data。再同時,我在template 中也有明確引用了這個data,那麼此時,這個data 的dep 裡就關聯了三個watcher,一個是render function 的watcher,一個是computed 的watcher,一個是用戶自己調用$ watch 方法所建立的watcher。當 data 發生變更後,這個 data 的 dep 就會通知到這三個 watcher 做出對應處理。

Observer

Observer 可以把一個 plainObject 或 array 變成 reactive 的。程式碼很少,就是遍歷 plainObject 或 array,對每一個鍵值呼叫

defineReactive 方法。

流程

以上三個類別介紹完了,基本上對 vue reactive 的實作應該有個模糊的認識,接下來,就結合實例講一下整個流程。

在 vue 實例化的時候,會先呼叫 initData,再呼叫 initComputed,最後再呼叫 mountComponent 建立 render function 的 watcher。從而完成一個 VueComponent 的資料 reactive 化。

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>
登入後複製

初始化流程:

  1. 将 name 处理为 reactive,创建 dep 实例
  2. 将 test 绑到 vm,创建 test 的 watcher 实例 watch1,添加 getter 逻辑。
  3. 创建 render function 的 watcher 实例 watcher2,并且立即执行 render function。
  4. 执行 render function 的时候,触发到 test 的 getter 逻辑,watcher1 及 watcher2 均与 dep 创建映射关系。

name 的值变更后的更新流程:

  1. 遍历绑定的 watcher 列表,执行 watcher.update()。
  2. watcher1.dirty 置为为 true。
  3. watcher2 重新执行 render function,触发到 test 的 getter,因为 watcher1.dirty 为 true,因此重新计算 test 的值,test 的值更新。
  4. 重渲染 view

【相关推荐:《vue.js教程》】

以上是解讀Vue之怎樣把資料包裝成reactive從而達到MDV效果的詳細內容。更多資訊請關注PHP中文網其他相關文章!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

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

AI Clothes Remover

AI Clothes Remover

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

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

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

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

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

熱門話題

Java教學
1658
14
CakePHP 教程
1415
52
Laravel 教程
1309
25
PHP教程
1257
29
C# 教程
1231
24
vue中怎麼用bootstrap vue中怎麼用bootstrap Apr 07, 2025 pm 11:33 PM

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

vue怎麼給按鈕添加函數 vue怎麼給按鈕添加函數 Apr 08, 2025 am 08:51 AM

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

vue中的watch怎麼用 vue中的watch怎麼用 Apr 07, 2025 pm 11:36 PM

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

vue多頁面開發是啥意思 vue多頁面開發是啥意思 Apr 07, 2025 pm 11:57 PM

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

vue返回上一頁的方法 vue返回上一頁的方法 Apr 07, 2025 pm 11:30 PM

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

vue遍歷怎麼用 vue遍歷怎麼用 Apr 07, 2025 pm 11:48 PM

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

React與Vue:Netflix使用哪個框架? React與Vue:Netflix使用哪個框架? Apr 14, 2025 am 12:19 AM

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

vue.js怎麼引用js文件 vue.js怎麼引用js文件 Apr 07, 2025 pm 11:27 PM

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

See all articles