vue data 值如何渲染-ag凯发k8国际
引言
相信绝大多数的前端小伙伴已记不清做了多少项目,写了多少代码了,每个人如同教科书般地写着vue代码:
// 单文件组件中常见代码 export default {data () {return {msg: 'click me'}},methods: {say () {this.msg = 'well done'}} }// 入口文件中的常见代码 new vue({el: '#app',router: router,render: h => h(app) })一切都显得那么自然。不过在百忙之中是否有小伙伴想过,一个小小的vue实例怎么有这么大的能量,竟然可以构建出如此复杂的前端项目。那么vue内部是如何运转的呢,做了哪些事情呢,从今天开始跟着我一探究竟。
vue是可以运行在多平台上的如浏览器,weex等,本文只分析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) }从vue的构造函数中可以看到,当我们执行new vue()的时候,只执行了一个_init方法。_init会根据传入的选项对vue进行初始化。props、data、生命周期,事件机制的初始化都是在此过程中完成的。
以data的初始化为例,vue会通过 object.defineproperty 的方式将data的属性定义到vue实例上。这也就解释了为什么我们可以在vue中通过对 this.msg 进行赋值,可以修改data中属性的值了。
以上对data的处理只是刚刚开始。为了能实现所谓的响应式或者数据驱动更新,vue又做了进一步的处理,具体做法是,创建一个observer对象,该对象与data绑定,通过 object.defineproperty 将data中的所有的属性转换成getter/setter。当data中的属性在vue实例中被访问(会触发getter),observer 对象就会把该属性收集为watcher实例的依赖,之后当data中的属性在vue实例中被改变(会触发setter), observer 会通知依赖该属性的 watcher 实例重新渲染页面。
注:每个watcher都对应一个vue实例以上处理流程串在一起,vue就实现了通过修改 this.msg 从而触发页面的自动更新。
最后借用vueag凯发k8国际官网上的一张示意图帮助大家再理解下这个处理过程:
模板解析
通过上面的分析,我们已经知道当数据发生变化时,会触发页面的重新渲染。接下来我们分析下vue是如何进行渲染的。
首先,vue会把将我们编写的html模板解析成一个ast描述对象,该对象是通过children和parent链接而成的树形结构,完整地描述了html标签的所有信息。
例如有如下html模板:
{{msg}}
最终会解析成如下形式的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: 19tag: "p",type: 1}],end: 263,parent: undefined,plain: false,rawattrsmap:{id: {name: "id", value: "app", start: 5, end: 13}},start: 0tag: "div",type: 1 }然后 vue 根据ast对象生成 render 函数,该函数的函数体大致如下:
with(this){return _c('div', {attrs:{"id":"app"}}, [_c('p', [_v(_s(msg))])]) }也就是说,我们的模板最终在vue内部都是会以一个render函数的形式存在。
vueag凯发k8国际官网上对此也有提及,一般推荐大家使用template,el等方式来指定模板,此外还可以通过使用render来自定义个性化的编译函数,不过vue内部最终都会解析成render函数。先虚后实
我们得到render函数之后,vue并未直接渲染成dom树,而是先通过render函数得到一个vnode。实际上这一步是非常有必要的,我们都知道频繁大量地操作dom节点是极耗性能的。vue在渲染之前通过对vnode的比较,可以大大规避非必要的dom操作。下面是一个vnode大致结构:
{tag: "div",children: [{tag: "p", ...}],data: {attrs: {id: "app"}}elm: dom节点(div#app),parent: undefined,context: vue实例,... }最后,vue根据diff之后的结果,执行真正的dom节点的插入更新删除等操作,同时触发vue实例的生命周期钩子函数。之后,vue要做的就是观察数据的变化,进而决定是否重新渲染页面了。
总结
以上就是vue在初始渲染过程中的主干流程,大体总结起来就是先对选项对象初始化,通过object.defineproperty建立一套响应式系统,然后将模板解析成render函数,然后使用render函数生成vnode,在渲染前,对vnode进行diff操作,最后进行必要的渲染。
本文并未深入每个执行中的代码细节,接下来会详细对初始化、响应式实现原理、模板渲染、指令解析、vnode的diff等进行介绍,敬请期待。
推荐阅读
- 如何使用 splitchunks 精细控制代码分割
- 用了这么久的react/vue,你真的了解hooks吗?
总结
以上是ag凯发k8国际为你收集整理的vue data 值如何渲染_vue执行流程解析的全部内容,希望文章能够帮你解决所遇到的问题。
- 上一篇: java判断括号是否闭合_【python
- 下一篇: 常用公差配合表图_涨知识!常用的机械测量