vue.js

关注公众号 jb51net

关闭
首页 > 网络编程 > JavaScript > javascript类库 > vue.js > Vue渲染执行流程

Vue执行流程及渲染示例解析

作者:Skywang

这篇文章主要为大家介绍了Vue执行流程及渲染解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

正文

最近想对之前看过的vue一些较原理的一些东西进行总结,今天就谈谈vue实例创建到渲染的一个流程概述。说的不对希望可以补充评论。

相信绝大多数的前端小伙伴已记不清做了多少项目,写了多少代码了,每个人如同教科书般地写着Vue代码:

      // 入口文件中的常见代码
      new Vue({
        el: '#app',
        router: router,
        render: h => h(App)
      })

大家是否有想过Vue内部是如何运转的呢,做了哪些事情呢?怎么在界面中渲染处预期效果呢!接下来我们慢慢探究!

初始化

Vue的构造函数

// Vue构造函数
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)
}

通过上面的函数可以看出当我们执行new Vue()的时候,只执行了一个_init方法。_init会根据传入的选项对vue进行初始化。我们初始化data的时候,vue会通过 Object.defineProperty 的方式将data的属性定义到vue实例上。这也就解释了为什么我们可以在vue中通过this.name进行赋值,可以修改data中name属性的值了。

为了能实现的响应式动态变化数据,vue又做了处理,创建一个observer对象,该对象与data绑定,通过 Object.defineProperty 将data中的所有的属性转换成getter/setter。当data中的属性在vue实例中被访问(会触发getter),observer 对象就会把该属性收集为watcher实例的依赖,之后当data中的属性在vue实例中被改变(会触发setter), observer 会通知依赖该属性的 watcher 实例重新渲染页面。

vue官网上的一张示意图帮助大家再理解下这个处理过程:

上面我们分析了vue是如和做到数据更新的,接下来我们看看他是如何做到渲染界面的。首先,vue会把将我们编写的HTML模板解析成一个AST描述对象,该对象是通过children和parent链接而成的树形结构,完整地描述了HTML标签的所有信息。

HTML模板

      <div id="app">
          <p>{{msg}}</p>
      </div>

最终会解析成下面这种AST对象

{
   attrs: [{name: "id", value: ""app"", dynamic: undefined, start: 5, end: 13}],
   attrsList: [{name: "id", value: "app", start: 5, end: 13}],
   attrsMap: {id: "app"},
   children: [{
        attrsList: [],
        attrsMap: {},
        children: [],
        end: 33,
        parent: {type: 1, tag: "div", ...},
        plain: true,
        pre: undefined,
        rawAttrsMap:{},
        start: 19
        tag: "p",
        type: 1
   }],
   end: 263,
   parent: undefined,
   plain: false,
   rawAttrsMap:{id: {name: "id", value: "app", start: 5, end: 13}},
   start: 0
   tag: "div",
   type: 1
}

然后 vue 根据AST对象生成 render 函数,该函数的函数体大致如下:

with(this){
    return _c('div', {attrs:{"id":"app"}}, [_c('p', [_v(_s(msg))])])
}

也就是说,我们的模板最终在vue内部都是会以一个render函数的形式存在。

函数 _c 是在初始化render环境的时候添加到vue实例上,用来创建 vnode 的全局实例方法。它可以通vue实例直接调用,主要是给vue内部使用的vnode创建方法。
我们得到render函数之后,vue并未直接渲染成DOM树,而是先通过render函数得到一个vnode。实际上这一步是非常有必要的,我们都知道频繁大量地操作DOM节点是极耗性能的。vue在渲染之前通过对vnode的比较,可以大大规避非必要的DOM操作。下面是一个vnode大致结构:

{
    tag: "div", // 元素标签,如div
    children: [{tag: "p", ...}], // vnode 子节点数组
    data: {attrs: {id: "app"}}, // 数据对象例如,{attrs: {id: 'app'}}
    elm: DOM节点(div#app),// 所对应的dom节点
    parent: undefined, // 父节点vnode
    context: Vue实例,  // 所对应的vue实例
    ...
}

方法 _v 也是vue实例方法,内部用以创建文本类型的vnode,在本例中,{{msg}}是一个文本节点,所以需要使用 _v 来创建文本vnode。不过无论是文本类型的vnode还是非文本类型的vnode都是Vnode对象的实例。两者的区别在于,文本类型的vnode不存在 tag 和 children。

// 创建一个文本类型的VNode
function createTextVNode (val) {
  return new VNode(undefined, undefined, undefined, String(val))
}

方法 _s 同样也是vue的实例方法,内部用来将接收的参数变成字符串返回,对于字符串和数值使用 Object.toString() 转换,如果接收到的是一个对象,则使用 JSON.stringify()转换。

function toString (val){
  return val == null
    ? ''
    : Array.isArray(val) || (isPlainObject(val) &amp;&amp; val.toString === Object.prototype.toString)
      ? JSON.stringify(val, null, 2)
      : String(val)
}

vnode 通过 parent 和 children 连接父节点和子节点,组成vnode树。最后,vue根据diff之后的结果,执行真正的dom节点的插入更新删除等操作,同时触发vue实例的生命周期钩子函数。之后,vue要做的就是观察数据的变化,进而决定是否重新渲染页面了。

继续分析vue是如何进行渲染的

创建DOM节点

有了vnode后,vue还需要根据vnode来创建DOM节点。如果是首次渲染,那么vue会走创建的逻辑。如果是数据的更新导致的重新渲染,那么vue会走更新的逻辑。

首次渲染

因为是首次渲染,所以不存在先前老的vnode,因此无需进行比较。vue直接调用 createElm 方法创建DOM元素。具体的创建步骤如下:

1.首先为vnode创建DOM元素。

2.如果vnode有子节点,逐个为其子节点创建DOM元素,并将子DOM元素插入到vnode的DOM元素上。

3.调用setAttribute 为vnode的DOM元素添加属性。

4.将vnode的DOM元素插入到其父元素上。

重新渲染

如果不是首次渲染,而是由数据变化所触发的重新渲染,那么vue会最大限度地复用已创建的DOM元素。而复用的前提就是通过比较新老vnode,找出需要更新的内容,然后最小限度地进行替换。这也是vue设计vnode的核心用途。vue源码中可以看到(此处先忽略),当新老vnode完全相等的情况下,vue不会对该节点重新渲染,直接跳过了。

如果新vnode发生了变化,那么vue会遵循以下步骤更新DOM元素:  

1.更新DOM元素的属性。  

这个在首次渲染那部分提到了一些。vue内实现了若干个属性处理模块,专门用于DOM元素属性的创建和更新。这些模块中基本都实现了create、update这两个处理函数。create 负责DOM元素属性的创建,update 负责DOM元素属性的更新。cbs.update[i](oldVnode, vnode) 的意思就是逐个调用这些模块上的 update 方法,以更新发生改变的DOM元素属性。

2.更新DOM元素的子元素。关于DOM子元素的更新分为几种情况

大量的DOM操作会极损耗浏览器性能。vue在每次数据发生变化后,都会重新生成vnode节点。通过比较新老vnode节点,找出需要进行操作的最小DOM元素子集。根据变化点,进行DOM元素属性、DOM子节点的更新。这种设计方式大大减少了DOM操作的次数

这次文章大部分都是看一些博客文章所了解的内容,基本上可以了解vue如何创建和如何渲染界面,还是老话好记性不如烂笔头 自己做了一些总结 可以加深理解!

更多关于Vue执行流程及渲染解析的资料请关注脚本之家其它相关文章!

您可能感兴趣的文章:
阅读全文