欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Vue指令工作原理實現(xiàn)方法

 更新時間:2021年06月28日 09:31:30   作者:Gerryli  
自定義指令是 vue 中使用頻率僅次于組件,其包含 bind 、 inserted 、 update 、 componentUpdated 、 unbind 五個生命周期鉤子,接下來通過本文給大家詳細(xì)介紹Vue指令工作原理實現(xiàn)方法,需要的朋友參考下吧

Vue簡介

現(xiàn)在的大前端時代,是一個動蕩紛爭的時代,江湖中已經(jīng)分成了很多門派,主要以Vue,React還有Angular為首,形成前端框架三足鼎立的局勢。Vue在前端框架中的地位就像曾經(jīng)的jQuery,由于其簡單易懂、開發(fā)效率高,已經(jīng)成為了前端工程師必不可少的技能之一。

Vue是一種漸進(jìn)式JavaScript框架,完美融合了第三方插件和UI組件庫,它和jQuery最大的區(qū)別在于,Vue無需開發(fā)人員直接操作DOM節(jié)點(diǎn),就可以改變頁面渲染內(nèi)容,在應(yīng)用開發(fā)者具有一定的HTML、CSS、JavaScript的基礎(chǔ)上,能夠快速上手,開發(fā)出優(yōu)雅、簡潔的應(yīng)用程序模塊。

前言

自定義指令是 vue 中使用頻率僅次于組件,其包含 bind 、 inserted 、 updatecomponentUpdated 、 unbind 五個生命周期鉤子。本文將對 vue 指令的工作原理進(jìn)行相應(yīng)介紹,從本文中,你將得到:

  • 指令的工作原理
  • 指令使用的注意事項

基本使用

官網(wǎng)案例:

<div id='app'>
  <input type="text" v-model="inputValue" v-focus>
</div>
<script>
  Vue.directive('focus', {
    // 第一次綁定元素時調(diào)用
    bind () {
      console.log('bind')
    },
    // 當(dāng)被綁定的元素插入到 DOM 中時……
    inserted: function (el) {
      console.log('inserted')
      el.focus()
    },
    // 所在組件VNode發(fā)生更新時調(diào)用
    update () {
      console.log('update')
    },
    // 指令所在組件的 VNode 及其子 VNode 全部更新后調(diào)用
    componentUpdated () {
      console.log('componentUpdated')
    },
    // 只調(diào)用一次,指令與元素解綁時調(diào)用
    unbind () {
      console.log('unbind')
    }
  })
  new Vue({
    data: {
      inputValue: ''
    }
  }).$mount('#app')
</script>

指令工作原理

初始化

初始化全局 API 時,在 platforms/web 下,調(diào)用 createPatchFunction 生成 VNode 轉(zhuǎn)換為真實 DOM 的 patch 方法,初始化中比較重要一步是定義了與 DOM 節(jié)點(diǎn)相對應(yīng)的 hooks 方法,在 DOM 的創(chuàng)建( create )、激活( avtivate )、更新( update )、移除( remove )、銷毀( destroy )過程中,分別會輪詢調(diào)用對應(yīng)的 hooks 方法,這些 hooks 中一部分是指令聲明周期的入口。

// src/core/vdom/patch.js
const hooks = ['create', 'activate', 'update', 'remove', 'destroy']
export function createPatchFunction (backend) {
  let i, j
  const cbs = {}

  const { modules, nodeOps } = backend
  for (i = 0; i < hooks.length; ++i) {
    cbs[hooks[i]] = []
    // modules對應(yīng)vue中模塊,具體有class, style, domListener, domProps, attrs, directive, ref, transition
    for (j = 0; j < modules.length; ++j) {
      if (isDef(modules[j][hooks[i]])) {
        // 最終將hooks轉(zhuǎn)換為{hookEvent: [cb1, cb2 ...], ...}形式
        cbs[hooks[i]].push(modules[j][hooks[i]])
      }
    }
  }
  // ....
  return function patch (oldVnode, vnode, hydrating, removeOnly) {
    // ...
  }
}

模板編譯

模板編譯就是解析指令參數(shù),具體解構(gòu)后的 ASTElement 如下所示:

{
  tag: 'input',
  parent: ASTElement,
  directives: [
    {
      arg: null, // 參數(shù)
      end: 56, // 指令結(jié)束字符位置
      isDynamicArg: false, // 動態(tài)參數(shù),v-xxx[dynamicParams]='xxx'形式調(diào)用
      modifiers: undefined, // 指令修飾符
      name: "model",
      rawName: "v-model", // 指令名稱
      start: 36, // 指令開始字符位置
      value: "inputValue" // 模板
    },
    {
      arg: null,
      end: 67,
      isDynamicArg: false,
      modifiers: undefined,
      name: "focus",
      rawName: "v-focus",
      start: 57,
      value: ""
    }
  ],
  // ...
}

生成渲染方法

vue 推薦采用指令的方式去操作 DOM ,由于自定義指令可能會修改 DOM 或者屬性,所以避免指令對模板解析的影響,在生成渲染方法時,首先處理的是指令,如 v-model ,本質(zhì)是一個語法糖,在拼接渲染函數(shù)時,會給元素加上 value 屬性與 input 事件(以 input 為例,這個也可以用戶自定義)。

with (this) {
    return _c('div', {
        attrs: {
            "id": "app"
        }
    }, [_c('input', {
        directives: [{
            name: "model",
            rawName: "v-model",
            value: (inputValue),
            expression: "inputValue"
        }, {
            name: "focus",
            rawName: "v-focus"
        }],
        attrs: {
            "type": "text"
        },
        domProps: {
            "value": (inputValue) // 處理v-model指令時添加的屬性
        },
        on: {
            "input": function($event) { // 處理v-model指令時添加的自定義事件
                if ($event.target.composing)
                    return;
                inputValue = $event.target.value
            }
        }
    })])
}

生成VNode

vue 的指令設(shè)計是方便我們操作 DOM ,在生成 VNode 時,指令并沒有做額外處理。

生成真實DOM

在 vue 初始化過程中,我們需要記住兩點(diǎn):

  • 狀態(tài)的初始化是 父 -> 子,如 beforeCreate 、 created 、 beforeMount ,調(diào)用順序是 父 -> 子
  • 真實 DOM 掛載順序是 子 -> 父,如 mounted ,這是因為在生成真實 DOM 過程中,如果遇到組件,會走組件創(chuàng)建的過程,真實 DOM 的生成是從子到父一級級拼接。

在 patch 過程中,每此調(diào)用 createElm 生成真實 DOM 時,都會檢測當(dāng)前 VNode 是否存在 data 屬性,存在,則會調(diào)用 invokeCreateHooks ,走初創(chuàng)建的鉤子函數(shù),核心代碼如下:

// src/core/vdom/patch.js
function createElm (
    vnode,
    insertedVnodeQueue,
    parentElm,
    refElm,
    nested,
    ownerArray,
    index
  ) {
    // ...
    // createComponent有返回值,是創(chuàng)建組件的方法,沒有返回值,則繼續(xù)走下面的方法
    if (createComponent(vnode, insertedVnodeQueue, parentElm, refElm)) {
      return
    }

    const data = vnode.data
    // ....
    if (isDef(data)) {
        // 真實節(jié)點(diǎn)創(chuàng)建之后,更新節(jié)點(diǎn)屬性,包括指令
        // 指令首次會調(diào)用bind方法,然后會初始化指令后續(xù)hooks方法
        invokeCreateHooks(vnode, insertedVnodeQueue)
    }
    // 從底向上,依次插入
    insert(parentElm, vnode.elm, refElm)
    // ...
  }

以上是指令鉤子方法的第一個入口,是時候揭露 directive.js 神秘的面紗了,核心代碼如下:

// src/core/vdom/modules/directives.js

// 默認(rèn)拋出的都是updateDirectives方法
export default {
  create: updateDirectives,
  update: updateDirectives,
  destroy: function unbindDirectives (vnode: VNodeWithData) {
    // 銷毀時,vnode === emptyNode
    updateDirectives(vnode, emptyNode)
  }
}

function updateDirectives (oldVnode: VNodeWithData, vnode: VNodeWithData) {
  if (oldVnode.data.directives || vnode.data.directives) {
    _update(oldVnode, vnode)
  }
}

function _update (oldVnode, vnode) {
  const isCreate = oldVnode === emptyNode
  const isDestroy = vnode === emptyNode
  const oldDirs = normalizeDirectives(oldVnode.data.directives, oldVnode.context)
  const newDirs = normalizeDirectives(vnode.data.directives, vnode.context)
  // 插入后的回調(diào)
  const dirsWithInsert = [
  // 更新完成后回調(diào)
  const dirsWithPostpatch = []

  let key, oldDir, dir
  for (key in newDirs) {
    oldDir = oldDirs[key]
    dir = newDirs[key]
    // 新元素指令,會執(zhí)行一次inserted鉤子方法
    if (!oldDir) {
      // new directive, bind
      callHook(dir, 'bind', vnode, oldVnode)
      if (dir.def && dir.def.inserted) {
        dirsWithInsert.push(dir)
      }
    } else {
      // existing directive, update
      // 已經(jīng)存在元素,會執(zhí)行一次componentUpdated鉤子方法
      dir.oldValue = oldDir.value
      dir.oldArg = oldDir.arg
      callHook(dir, 'update', vnode, oldVnode)
      if (dir.def && dir.def.componentUpdated) {
        dirsWithPostpatch.push(dir)
      }
    }
  }

  if (dirsWithInsert.length) {
    // 真實DOM插入到頁面中,會調(diào)用此回調(diào)方法
    const callInsert = () => {
      for (let i = 0; i < dirsWithInsert.length; i++) {
        callHook(dirsWithInsert[i], 'inserted', vnode, oldVnode)
      }
    }
    // VNode合并insert hooks
    if (isCreate) {
      mergeVNodeHook(vnode, 'insert', callInsert)
    } else {
      callInsert()
    }
  }

  if (dirsWithPostpatch.length) {
    mergeVNodeHook(vnode, 'postpatch', () => {
      for (let i = 0; i < dirsWithPostpatch.length; i++) {
        callHook(dirsWithPostpatch[i], 'componentUpdated', vnode, oldVnode)
      }
    })
  }

  if (!isCreate) {
    for (key in oldDirs) {
      if (!newDirs[key]) {
        // no longer present, unbind
        callHook(oldDirs[key], 'unbind', oldVnode, oldVnode, isDestroy)
      }
    }
  }
}

對于首次創(chuàng)建,執(zhí)行過程如下:

  1. oldVnode === emptyNode , isCreate 為 true ,調(diào)用當(dāng)前元素中所有 bind 鉤子方法。
  2. 檢測指令中是否存在 inserted 鉤子,如果存在,則將 insert 鉤子合并到 VNode.data.hooks 屬性中。
  3. DOM 掛載結(jié)束后,會執(zhí)行 invokeInsertHook ,所有已掛載節(jié)點(diǎn),如果 VNode.data.hooks 中存在 insert 鉤子。則會調(diào)用,此時會觸發(fā)指令綁定的 inserted 方法。

一般首次創(chuàng)建只會走 bind inserted 方法,而 update componentUpdated 則與 bind 和 inserted 對應(yīng)。在組件依賴狀態(tài)發(fā)生改變時,會用 VNode diff 算法,對節(jié)點(diǎn)進(jìn)行打補(bǔ)丁式更新,其調(diào)用流程:

  1. 響應(yīng)式數(shù)據(jù)發(fā)生改變,調(diào)用 dep.notify ,通知數(shù)據(jù)更新。
  2. 調(diào)用 patchVNode ,對新舊 VNode 進(jìn)行差異化更新,并全量更新當(dāng)前 VNode 屬性(包括指令,就會進(jìn)入 updateDirectives 方法)。
  3. 如果指令存在 update 鉤子方法,調(diào)用 update 鉤子方法,并初始化 componentUpdated 回調(diào),將 postpatch hooks 掛載到 VNode.data.hooks 中。
  4. 當(dāng)前節(jié)點(diǎn)及子節(jié)點(diǎn)更新完畢后,會觸發(fā) postpatch hooks ,即指令的 componentUpdated 方法

核心代碼如下:

// src/core/vdom/patch.js
function patchVnode (
    oldVnode,
    vnode,
    insertedVnodeQueue,
    ownerArray,
    index,
    removeOnly
  ) {
    // ...
    const oldCh = oldVnode.children
    const ch = vnode.children
    // 全量更新節(jié)點(diǎn)的屬性
    if (isDef(data) && isPatchable(vnode)) {
      for (i = 0; i < cbs.update.length; ++i) cbs.update[i](oldVnode, vnode)
      if (isDef(i = data.hook) && isDef(i = i.update)) i(oldVnode, vnode)
    }
    // ...
    if (isDef(data)) {
    // 調(diào)用postpatch鉤子
      if (isDef(i = data.hook) && isDef(i = i.postpatch)) i(oldVnode, vnode)
    }
  }

unbind 方法是在節(jié)點(diǎn)銷毀時,調(diào)用 invokeDestroyHook ,這里不做過多描述。

注意事項

使用自定義指令時,和普通模板數(shù)據(jù)綁定, v-model 還是存在一定的差別,如雖然我傳遞參數(shù)( v-xxx='param' )是一個引用類型,數(shù)據(jù)變化時,并不能觸發(fā)指令的 bind 或者 inserted ,這是因為在指令的聲明周期內(nèi), bind 和 inserted 只是在初始化時調(diào)用一次,后面只會走 update componentUpdated 。

指令的聲明周期執(zhí)行順序為 bind -> inserted -> update -> componentUpdated ,如果指令需要依賴于子組件的內(nèi)容時,推薦在 componentUnpdated 中寫相應(yīng)業(yè)務(wù)邏輯。

vue 中,很多方法都是循環(huán)調(diào)用,如 hooks 方法,事件回調(diào)等,一般調(diào)用都用 try catch 包裹,這樣做的目的是為了防止一個處理方法報錯,導(dǎo)致整個程序崩潰,這一點(diǎn)在我們開發(fā)過程中可以借鑒使用。

小結(jié)

開始看整個 vue 源碼時,對很多細(xì)枝末節(jié)方法都不怎么了解,通過梳理具體每個功能的實現(xiàn)時,漸漸能夠看到整個 vue 全貌,同時也能避免開發(fā)使用中的一些坑點(diǎn)。

GitHub

以上就是Vue指令工作原理實現(xiàn)方法的詳細(xì)內(nèi)容,更多關(guān)于Vue指令原理的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • vue3.x項目降級到vue2.7的解決方案

    vue3.x項目降級到vue2.7的解決方案

    Vue2.7是Vue2.x的最終次要版本,下面這篇文章主要給大家介紹了關(guān)于vue3.x項目降級到vue2.7的解決方案,文中通過實例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2023-03-03
  • 解決Vue打包上線之后部分CSS不生效的問題

    解決Vue打包上線之后部分CSS不生效的問題

    今天小編就為大家分享一篇解決Vue打包上線之后部分CSS不生效的問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2019-11-11
  • vue-router 4使用實例詳解

    vue-router 4使用實例詳解

    雖然 vue-router 4 大多數(shù) API 保持不變,但是在 vue3 中以插件形式存在,所以在使用時有一定的變化。接下來就學(xué)習(xí)學(xué)習(xí)它是如何使用的
    2021-11-11
  • Vue3.0中Ref與Reactive的區(qū)別示例詳析

    Vue3.0中Ref與Reactive的區(qū)別示例詳析

    在vue3中對響應(yīng)式數(shù)據(jù)的聲明官方給出了ref()和reactive()這兩種方式,這篇文章主要給大家介紹了關(guān)于Vue3.0中Ref與Reactive區(qū)別的相關(guān)資料,需要的朋友可以參考下
    2021-07-07
  • Vite多環(huán)境配置項目高定制化能力詳解

    Vite多環(huán)境配置項目高定制化能力詳解

    這篇文章主要為大家介紹了Vite多環(huán)境配置項目高定制化能力詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-07-07
  • Vue.js 表單校驗插件

    Vue.js 表單校驗插件

    這篇文章主要介紹了Vue.js 表單校驗插件的相關(guān)資料,需要的朋友可以參考下
    2016-08-08
  • vue路由事件beforeRouteLeave及組件內(nèi)定時器的清除方法

    vue路由事件beforeRouteLeave及組件內(nèi)定時器的清除方法

    今天小編就為大家分享一篇vue路由事件beforeRouteLeave及組件內(nèi)定時器的清除方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-09-09
  • Vue簡易版無限加載組件實現(xiàn)原理與示例代碼

    Vue簡易版無限加載組件實現(xiàn)原理與示例代碼

    這篇文章主要給大家介紹了關(guān)于Vue簡易版無限加載組件實現(xiàn)原理與示例代碼的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家學(xué)習(xí)或者使用vue具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2022-07-07
  • Vuejs仿網(wǎng)易云音樂實現(xiàn)聽歌及搜索功能

    Vuejs仿網(wǎng)易云音樂實現(xiàn)聽歌及搜索功能

    這篇文章主要介紹了Vuejs仿網(wǎng)易云音樂實現(xiàn)聽歌及搜索功能,非常不錯,具有參考借鑒價值,需要的朋友可以參考下
    2017-03-03
  • vue中使用codemirror的實例詳解

    vue中使用codemirror的實例詳解

    這篇文章主要介紹了vue中使用codemirror的實例教程,非常不錯,具有一定的參考借鑒價值 ,需要的朋友可以參考下
    2018-11-11

最新評論