成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

【Vue源碼學(xué)習(xí)】vue實(shí)例化到掛載到dom(上)

Hegel_Gu / 1233人閱讀

摘要:作者王聰本篇目的是介紹實(shí)例化到掛載到的整體路線,一些細(xì)節(jié)會被省略。從源碼中找到構(gòu)造函數(shù)的聲明,是一個(gè)很簡潔的工廠模式聲明的一個(gè)構(gòu)造函數(shù)。內(nèi)部做了邏輯判斷構(gòu)造函數(shù)調(diào)用必須有關(guān)鍵字。代表的是當(dāng)前實(shí)例也就是構(gòu)造函數(shù)被調(diào)用后的指向。

作者:王聰
本篇目的是介紹vue實(shí)例化到掛載到dom的整體路線,一些細(xì)節(jié)會被省略。

從new Vue()開始

所有的一切都是從 new Vue()開始的,所以從這個(gè)點(diǎn)開始探尋這個(gè)過程發(fā)生了什么。

從源碼中找到Vue構(gòu)造函數(shù)的聲明,src/core/instance/index.js

import { initMixin } from "./init"
import { stateMixin } from "./state"
import { renderMixin } from "./render"
import { eventsMixin } from "./events"
import { lifecycleMixin } from "./lifecycle"
import { warn } from "../util/index"

function Vue (options) {
  if (process.env.NODE_ENV !== "production" &&
    !(this instanceof Vue)
  ) {
    warn("Vue is a constructor and should be called with the `new` keyword")
  }
  this._init(options)
}

initMixin(Vue)
stateMixin(Vue)
eventsMixin(Vue)
lifecycleMixin(Vue)
renderMixin(Vue)

export default Vue

是一個(gè)很簡潔的function工廠模式聲明的一個(gè)構(gòu)造函數(shù)。
內(nèi)部做了邏輯判斷:構(gòu)造函數(shù)調(diào)用必須有 new 關(guān)鍵字。
然后執(zhí)行了this._init(options),該初始化函數(shù)就是Vue 初始化的開始。

Vue.prototype._init()

this._init()是在何時(shí)聲明的呢?通過下邊5個(gè)初始化函數(shù)的執(zhí)行,在Vue原型鏈中添加了大量的的屬性與函數(shù)。this._init()實(shí)際就是調(diào)用了原型鏈上的Vue.prototype._init()函數(shù)。

initMixin(Vue)
stateMixin(Vue)
eventsMixin(Vue)
lifecycleMixin(Vue)
renderMixin(Vue)

Vue.prototype._init函數(shù)是在initMixin(Vue)中去添加到原型鏈上的。在src/core/instance/init.js中定義

Vue.prototype._init = function (options?: Object) {
    const vm: Component = this
    // a uid
    vm._uid = uid++

    let startTag, endTag
    /* istanbul ignore if */
    if (process.env.NODE_ENV !== "production" && config.performance && mark) {
      startTag = `vue-perf-start:${vm._uid}`
      endTag = `vue-perf-end:${vm._uid}`
      mark(startTag)
    }

    // a flag to avoid this being observed
    vm._isVue = true
    // merge options
    if (options && options._isComponent) {
      // optimize internal component instantiation
      // since dynamic options merging is pretty slow, and none of the
      // internal component options needs special treatment.
      initInternalComponent(vm, options)
    } else {
      vm.$options = mergeOptions(
        resolveConstructorOptions(vm.constructor),
        options || {},
        vm
      )
    }
    /* istanbul ignore else */
    if (process.env.NODE_ENV !== "production") {
      initProxy(vm)
    } else {
      vm._renderProxy = vm
    }
    // expose real self
    vm._self = vm
    initLifecycle(vm)
    initEvents(vm)
    initRender(vm)
    callHook(vm, "beforeCreate")
    initInjections(vm) // resolve injections before data/props
    initState(vm)
    initProvide(vm) // resolve provide after data/props
    callHook(vm, "created")

    /* istanbul ignore if */
    if (process.env.NODE_ENV !== "production" && config.performance && mark) {
      vm._name = formatComponentName(vm, false)
      mark(endTag)
      measure(`vue ${vm._name} init`, startTag, endTag)
    }

    if (vm.$options.el) {
      vm.$mount(vm.$options.el)
    }
  }
}

_init函數(shù)內(nèi)部又通過上邊的這種調(diào)用初始化函數(shù)的模式完成了具體模塊的初始化。聲明鉤子的調(diào)用也首次在這里出現(xiàn),通過分析這些函數(shù)調(diào)用順序,能更好的理解官方文檔中提及的各個(gè)生命周期鉤子函數(shù)的觸發(fā)時(shí)機(jī)。

initLifecycle(vm)
initEvents(vm)
initRender(vm)
callHook(vm, "beforeCreate")
initInjections(vm) // resolve injections before data/props
initState(vm)
initProvide(vm) // resolve provide after data/props
callHook(vm, "created")

_init()函數(shù)的最后,執(zhí)行了vm.$mount()。vm代表的是當(dāng)前vue實(shí)例也就是構(gòu)造函數(shù)被調(diào)用后的this指向。
**$mount**是何時(shí)在vue上聲明的呢?這次并不是在上邊的初始化函數(shù)中完成聲明的。因?yàn)?$mount 這個(gè)方法的實(shí)現(xiàn)是和平臺、構(gòu)建方式都相關(guān),所以在不同構(gòu)建入口文件中有不同的定義。

Vue.prototype.$mount

原型上聲明的 $mount方法在 src/platform/web/runtime/index.js 中定義,這個(gè)方法會被runtime only版本和含編譯器完整版中復(fù)用,vue版本說明。

// public mount method
Vue.prototype.$mount = function (
  el?: string | Element,
  hydrating?: boolean
): Component {
  el = el && inBrowser ? query(el) : undefined
  return mountComponent(this, el, hydrating)
}

現(xiàn)在接著上邊Vue.prototype._init函數(shù)中調(diào)用了vm.$mount函數(shù),實(shí)際上是執(zhí)行了mountComponent(this, el, hydrating)函數(shù)。
mountComponent定義在目錄src/core/instance/lifecycle.js

export function mountComponent (
  vm: Component,
  el: ?Element,
  hydrating?: boolean
): Component {
  vm.$el = el
  if (!vm.$options.render) {
    vm.$options.render = createEmptyVNode
    if (process.env.NODE_ENV !== "production") {
      /* istanbul ignore if */
      if ((vm.$options.template && vm.$options.template.charAt(0) !== "#") ||
        vm.$options.el || el) {
        warn(
          "You are using the runtime-only build of Vue where the template " +
          "compiler is not available. Either pre-compile the templates into " +
          "render functions, or use the compiler-included build.",
          vm
        )
      } else {
        warn(
          "Failed to mount component: template or render function not defined.",
          vm
        )
      }
    }
  }
  callHook(vm, "beforeMount")

  let updateComponent
  /* istanbul ignore if */
  if (process.env.NODE_ENV !== "production" && config.performance && mark) {
    updateComponent = () => {
      const name = vm._name
      const id = vm._uid
      const startTag = `vue-perf-start:${id}`
      const endTag = `vue-perf-end:${id}`

      mark(startTag)
      const vnode = vm._render()
      mark(endTag)
      measure(`vue ${name} render`, startTag, endTag)

      mark(startTag)
      vm._update(vnode, hydrating)
      mark(endTag)
      measure(`vue ${name} patch`, startTag, endTag)
    }
  } else {
    updateComponent = () => {
      vm._update(vm._render(), hydrating)
    }
  }

  // we set this to vm._watcher inside the watcher"s constructor
  // since the watcher"s initial patch may call $forceUpdate (e.g. inside child
  // component"s mounted hook), which relies on vm._watcher being already defined
  new Watcher(vm, updateComponent, noop, {
    before () {
      if (vm._isMounted && !vm._isDestroyed) {
        callHook(vm, "beforeUpdate")
      }
    }
  }, true /* isRenderWatcher */)
  hydrating = false

  // manually mounted instance, call mounted on self
  // mounted is called for render-created child components in its inserted hook
  if (vm.$vnode == null) {
    vm._isMounted = true
    callHook(vm, "mounted")
  }
  return vm
}

拋開源碼中對性能測試和異常警告部分的代碼,這個(gè)函數(shù)內(nèi)部做了以下事情:

callHook(vm, "beforeMount"),調(diào)用生命周期鉤子beforeMount

聲明了updateComponent函數(shù)

new Watcher (vm, updateComponent, noop, {... callHook(vm, "beforeUpdate")}})

callHook(vm, "mounted"),調(diào)用生命周期鉤子mounted

這里的new Watcher()實(shí)例化了Watcher類,內(nèi)部邏輯先不去深入,這里僅僅需要知道的是在這個(gè)實(shí)例化的過程中調(diào)用了作為參數(shù)傳入的updateComponent函數(shù),而從這個(gè)函數(shù)的聲明來看,它實(shí)際上執(zhí)行的是vm._update(vm._render(), hydrating)這個(gè)函數(shù)。
首先vm._update和vm._render這兩個(gè)方法是定義在Vue原型上的。

Vue.prototype._render

Vue 的 _render 方法是用來把實(shí)例渲染成一個(gè)虛擬 Node。是在renderMixin(Vue)執(zhí)行時(shí)聲明的。它的定義在 src/core/instance/render.js 文件中

Vue.prototype._render = function (): VNode {
  const vm: Component = this
  const { render, _parentVnode } = vm.$options

  // reset _rendered flag on slots for duplicate slot check
  if (process.env.NODE_ENV !== "production") {
    for (const key in vm.$slots) {
      // $flow-disable-line
      vm.$slots[key]._rendered = false
    }
  }

  if (_parentVnode) {
    vm.$scopedSlots = _parentVnode.data.scopedSlots || emptyObject
  }

  // set parent vnode. this allows render functions to have access
  // to the data on the placeholder node.
  vm.$vnode = _parentVnode
  // render self
  let vnode
  try {
    vnode = render.call(vm._renderProxy, vm.$createElement)
  } catch (e) {
    handleError(e, vm, `render`)
    // return error render result,
    // or previous vnode to prevent render error causing blank component
    /* istanbul ignore else */
    if (process.env.NODE_ENV !== "production") {
      if (vm.$options.renderError) {
        try {
          vnode = vm.$options.renderError.call(vm._renderProxy, vm.$createElement, e)
        } catch (e) {
          handleError(e, vm, `renderError`)
          vnode = vm._vnode
        }
      } else {
        vnode = vm._vnode
      }
    } else {
      vnode = vm._vnode
    }
  }
  // return empty vnode in case the render function errored out
  if (!(vnode instanceof VNode)) {
    if (process.env.NODE_ENV !== "production" && Array.isArray(vnode)) {
      warn(
        "Multiple root nodes returned from render function. Render function " +
        "should return a single root node.",
        vm
      )
    }
    vnode = createEmptyVNode()
  }
  // set parent
  vnode.parent = _parentVnode
  return vnode
}

函數(shù)內(nèi)部對不同邏輯有不同的處理,但最終返回的都是VNode。
Virtual DOM 就是用一個(gè)原生的 JS 對象去描述一個(gè) DOM 節(jié)點(diǎn)。

Vue.prototype._update函數(shù)

_update是在lifecycleMixin(Vue)函數(shù)執(zhí)行時(shí)添加的。在目錄src/core/instance/lifecycle.js

Vue.prototype._update = function (vnode: VNode, hydrating?: boolean) {
    const vm: Component = this
    const prevEl = vm.$el
    const prevVnode = vm._vnode
    const restoreActiveInstance = setActiveInstance(vm)
    vm._vnode = vnode
    // Vue.prototype.__patch__ is injected in entry points
    // based on the rendering backend used.
    if (!prevVnode) {
      // initial render
      vm.$el = vm.__patch__(vm.$el, vnode, hydrating, false /* removeOnly */)
    } else {
      // updates
      vm.$el = vm.__patch__(prevVnode, vnode)
    }
    restoreActiveInstance()
    // update __vue__ reference
    if (prevEl) {
      prevEl.__vue__ = null
    }
    if (vm.$el) {
      vm.$el.__vue__ = vm
    }
    // if parent is an HOC, update its $el as well
    if (vm.$vnode && vm.$parent && vm.$vnode === vm.$parent._vnode) {
      vm.$parent.$el = vm.$el
    }
    // updated hook is called by the scheduler to ensure that children are
    // updated in a parent"s updated hook.
  }

通過源碼看接受的參數(shù):

vnode: VNode

hydrating?: boolean

接受的第一個(gè)參數(shù)類型是VNode(用來描述dom節(jié)點(diǎn)的虛擬節(jié)點(diǎn)),第二個(gè)布爾類型的參數(shù)是跟ssr相關(guān)。
函數(shù)內(nèi)部最關(guān)鍵的執(zhí)行了vm.$el = vm.__patch__(...)
調(diào)用 vm.__patch__ 去把 VNode 轉(zhuǎn)換成真正的 DOM 節(jié)點(diǎn)

現(xiàn)在回顧總結(jié)一下目前的流程:

new Vue()操作后,會調(diào)用Vue.prototype._init()方法。完成一系列初始化(原型上添加方法和屬性)

執(zhí)行Vue.prototype.$mount

vm._render()獲取描述當(dāng)前實(shí)例的VNode

vm._update(VNode),調(diào)用 vm.__patch__轉(zhuǎn)換成真正的 DOM 節(jié)點(diǎn)

流程圖:

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/103564.html

相關(guān)文章

  • vue 源碼學(xué)習(xí)(二) 實(shí)例初始化和掛載過程

    摘要:最后判斷有無根節(jié)點(diǎn),無則表示首次掛載,添加鉤子函數(shù),返回總結(jié)實(shí)例初始化掛載方法屬性初始化掛載過程在版本,生成函數(shù)對作處理,執(zhí)行中定義了通過實(shí)例化的回調(diào)執(zhí)行執(zhí)行,即調(diào)用了真實(shí)渲染成對象。 vue 入口 從vue的構(gòu)建過程可以知道,web環(huán)境下,入口文件在 src/platforms/web/entry-runtime-with-compiler.js(以Runtime + Compile...

    時(shí)飛 評論0 收藏0
  • Vue原理】Component - 源碼版 之 掛載組件DOM

    摘要:寫文章不容易,點(diǎn)個(gè)贊唄兄弟專注源碼分享,文章分為白話版和源碼版,白話版助于理解工作原理,源碼版助于了解內(nèi)部詳情,讓我們一起學(xué)習(xí)吧研究基于版本如果你覺得排版難看,請點(diǎn)擊下面鏈接或者拉到下面關(guān)注公眾號也可以吧原理源碼版之掛載組件由這篇文章從模 寫文章不容易,點(diǎn)個(gè)贊唄兄弟專注 Vue 源碼分享,文章分為白話版和 源碼版,白話版助于理解工作原理,源碼版助于了解內(nèi)部詳情,讓我們一起學(xué)習(xí)吧研究基于...

    lbool 評論0 收藏0
  • Vue原理】Component - 白話版

    摘要:寫文章不容易,點(diǎn)個(gè)贊唄兄弟專注源碼分享,文章分為白話版和源碼版,白話版助于理解工作原理,源碼版助于了解內(nèi)部詳情,讓我們一起學(xué)習(xí)吧研究基于版本如果你覺得排版難看,請點(diǎn)擊下面鏈接或者拉到下面關(guān)注公眾號也可以吧原理白話版從模板上使用到掛載到頁面 寫文章不容易,點(diǎn)個(gè)贊唄兄弟專注 Vue 源碼分享,文章分為白話版和 源碼版,白話版助于理解工作原理,源碼版助于了解內(nèi)部詳情,讓我們一起學(xué)習(xí)吧研究基于...

    liuyix 評論0 收藏0
  • Vue原理】從模板DOM的簡要流程

    摘要:寫文章不容易,點(diǎn)個(gè)贊唄兄弟專注源碼分享,文章分為白話版和源碼版,白話版助于理解工作原理,源碼版助于了解內(nèi)部詳情,讓我們一起學(xué)習(xí)吧研究基于版本如果你覺得排版難看,請點(diǎn)擊下面鏈接或者拉到下面關(guān)注公眾號也可以吧原理從模板到的簡要流程今天的計(jì)劃是, 寫文章不容易,點(diǎn)個(gè)贊唄兄弟專注 Vue 源碼分享,文章分為白話版和 源碼版,白話版助于理解工作原理,源碼版助于了解內(nèi)部詳情,讓我們一起學(xué)習(xí)吧研究基...

    wenzi 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<