Vue vm.$options 中的render

new Vue({
  el:'#app',
  render: h => h(App),
})

vm.$options指的是:

{
  el:'#app',
  render: h => h(App),
}

在Vue.prototype._render 代碼中我們可以看到:
var render = vm.$options.render;

實際上這個render指的就是上面實例化時的render函數(shù)

 var vm = this;
 var ref = vm.$options;
 var render = ref.render;
Vue.prototype._render = function () {

    var vm = this;
    var ref = vm.$options;
    var render = ref.render;

    var _parentVnode = ref._parentVnode;

    if (_parentVnode) {
      vm.$scopedSlots = normalizeScopedSlots(
        _parentVnode.data.scopedSlots,
        vm.$slots,
        vm.$scopedSlots
      );
    }

    // set parent vnode. this allows render functions to have access
    // to the data on the placeholder node.
    vm.$vnode = _parentVnode;
    // render self
    var vnode;
    try {
      // There's no need to maintain a stack because all render fns are called
      // separately from one another. Nested component's render fns are called
      // when parent component is patched.
      currentRenderingInstance = vm;
      vnode = render.call(vm._renderProxy, vm.$createElement);
      console.log('render ---- 999')
      console.log(vnode)
    } 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' && 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;
      }
    } finally {
      currentRenderingInstance = null;
    }
    // if the returned array contains only a single node, allow it
    if (Array.isArray(vnode) && vnode.length === 1) {
      vnode = vnode[0];
    }
    // 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
  };

了解下面的vnode的實現(xiàn):

 try {
      // There's no need to maintain a stack because all render fns are called
      // separately from one another. Nested component's render fns are called
      // when parent component is patched.
      currentRenderingInstance = vm;
      vnode = render.call(vm._renderProxy, vm.$createElement);
    }

現(xiàn)在數(shù)據(jù)流結(jié)構(gòu)是:

step1:入口文件是main.js

/*------------main.js------------*/
import Vue from 'vue'
import App from './App.vue'

Vue.config.productionTip = false

new Vue({
  el:'#app',
  render: h => h(App),
})

step2:組件App.vue

/*------------App.vue------------*/
<template>
  <div id="app1">
    <img alt="Vue logo" src="./assets/logo.png">
    <HelloWorld msg="Welcome to Your Vue.js App"/>
  </div>
</template>

<script>
import HelloWorld from './components/HelloWorld.vue'

export default {
  name: 'App',
  components: {
    HelloWorld
  }
}
</script>

step3:組件App.vue中的組件HelloWorld

/*------------HelloWorld.vue------------*/
<template>
  <div id="hello" class="hello">
    <h1>{{ msg }}</h1>
  </div>
</template>

<script>
export default {
  name: 'HelloWorld',
  props: {
    msg: String
  },
}
</script>

文件關(guān)數(shù)據(jù)流

image.png

然后在_render函數(shù)中我們可以看到vnode生成的數(shù)據(jù)對象是:

1. main.js

-context: Vue {_uid: 0
-parent: null

VNode {
asyncFactory: undefined
asyncMeta: undefined
children: undefined
componentInstance: VueComponent {_uid: 1, _isVue: true, $options: {…}, _renderProxy: Proxy, _self: VueComponent, …}
componentOptions: {propsData: undefined, listeners: undefined, tag: undefined, children: undefined, Ctor: ?}
context: Vue {_uid: 0, _isVue: true, $options: {…}, _renderProxy: Proxy, _self: Vue, …}
data: {on: undefined, hook: {…}, pendingInsert: null}
elm: div#app1
fnContext: undefined
fnOptions: undefined
fnScopeId: undefined
isAsyncPlaceholder: false
isCloned: false
isComment: false
isOnce: false
isRootInsert: true
isStatic: false
key: undefined
ns: undefined
parent: null
raw: false
tag: "vue-component-1-App"
text: undefined
child: (...)
__proto__: Object
}

2. App.vue

-context: Vue {_uid: 1
-parent: VNode {tag: "vue-component-1-App", data: {…}, children: undefined, text: undefined, elm: div#app1, …}

VNode {
asyncFactory: undefined
asyncMeta: undefined
children: (2) [VNode, VNode]
componentInstance: undefined
componentOptions: undefined
context: VueComponent {_uid: 1, _isVue: true, $options: {…}, _renderProxy: Proxy, _self: VueComponent, …}
data: {attrs: {…}}
elm: div#app1
fnContext: undefined
fnOptions: undefined
fnScopeId: undefined
isAsyncPlaceholder: false
isCloned: false
isComment: false
isOnce: false
isRootInsert: true
isStatic: false
key: undefined
ns: undefined
parent: VNode {tag: "vue-component-1-App", data: {…}, children: undefined, text: undefined, elm: div#app1, …}
raw: false
tag: "div"
text: undefined
child: (...)
__proto__: Object
}

3. HelloWorld.vue

-context: Vue {_uid: 2
-parent: VNode {tag: "vue-component-2-HelloWorld", data: {…}, children: undefined, text: undefined, elm: div#hello.hello, …}

VNode {
asyncFactory: undefined
asyncMeta: undefined
children: (8) [VNode, VNode, VNode, VNode, VNode, VNode, VNode, VNode]
componentInstance: undefined
componentOptions: undefined
context: VueComponent {_uid: 2, _isVue: true, $options: {…}, _renderProxy: Proxy, _self: VueComponent, …}
data: {staticClass: "hello", attrs: {…}}
elm: div#hello.hello
fnContext: undefined
fnOptions: undefined
fnScopeId: undefined
isAsyncPlaceholder: false
isCloned: false
isComment: false
isOnce: false
isRootInsert: true
isStatic: false
key: undefined
ns: undefined
parent: VNode {tag: "vue-component-2-HelloWorld", data: {…}, children: undefined, text: undefined, elm: div#hello.hello, …}
raw: false
tag: "div"
text: undefined
child: (...)
__proto__: Object
}

從上面的打印結(jié)果可以了解到render生成vnode的對象,及對應(yīng)的父子關(guān)系

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

友情鏈接更多精彩內(nèi)容