写点什么

Vuejs 2.0 源码解析之渲染篇

  • 2019-08-21
  • 本文字数:2376 字

    阅读完需:约 8 分钟

Vuejs 2.0源码解析之渲染篇

1、Vuejs 框架是目前比较火的 MVVM 框架之一,简单易上手的学习曲线,友好的官方文档,配套的构建工具,让 Vuejs 在 2016 大放异彩,大有赶超 react 之势。前不久 Vuejs 2.0 正式版已出,优化体积(相比 1.0 减少了 50%),提升性能(相比 1.0 提升 60%),API 优化等,各方面更上一层楼。

2、本文是系列文章,主要想通过对于 Vuejs 2.0 源码的分析,从代码层面解析 Vuejs 的实现原理,帮助读者能够更深入地理解整个框架的思想。此篇文章主要介绍前端渲染部分。

3、不足之处还请批评指正,欢迎一起交流学习。

Vue 的初始化

我们在使用 Vuejs 的时候,最基本的一个使用,就是在 HTML 引入 Vuejs 的库文件,并写如下一段代码:



new Vue,本质就是生成一个 Vue 的对象,我们来了解一下这个生成 Vue 对象的过程是怎样的。


1、首先,vue 的入口是/src/entries/web-runtime-with-compiler.js,这是由 config.js 配置文件决定的。



这个入口文件中 import 了很多文件,其中有一条主要的脉络:



其中/src/core/instance/index.js 是最核心的初始化代码,其中:



红框部分,就是整个 Vue 的类的核心方法。其含义给读者解读一下:



其中 new Vue 就是执行下面的这个函数:



_init 方法就是 initMixin 中的_init 方法



至此,程序沿着这个_init 方法继续走下去。

Vue 的渲染逻辑—render 函数

在定义完成 Vue 对象的初始化工作之后,本文主要是讲渲染部分,那么我们接上面的逻辑,看 Vuejs 是如何渲染页面的。在上图中我们看到有一个 initRender 的方法:



在该方法中会执行红框部分的内容:



而 $mount 方法就是整个渲染过程的起始点。具体定义是在/src/entries/web-runtime-with-compiler.js 中,根据代码整理成流程图:



由此图可以看到,在渲染过程中,提供了三种渲染模式,自定义 render 函数、template、el 均可以渲染页面,也就是对应我们使用 vue 时,三种写法:


1、自定义 render 函数



2、template 写法



3、el 写法(这个就是入门时最基本的写法)



这三种渲染模式最终都是要得到 render 函数。只不过用户自定义的 render 函数省去了程序分析的过程,等同于处理过的 render 函数,而普通的 template 或者 el 只是字符串,需要解析成 AST,再将 AST 转化为 render 函数。


记住一点,无论哪种方法,都要得到 render 函数。


我们在使用过程中具体要使用哪种调用方式,要根据具体的需求来。


1、如果是比较简单的逻辑,使用 template 和 el 比较好,因为这两种都属于声明式渲染,对用户理解比较容易,但灵活性比较差,因为最终生成的 render 函数是由程序通过 AST 解析优化得到的。


2、而使用自定义 render 函数相当于人已经将逻辑翻译给程序,能够胜任复杂的逻辑,灵活性高,但对于用户的理解相对差点。

Vue 的渲染逻辑—VNode 对象 &patch 方法

根据上面的结论,我们无论怎么渲染,最终会得到 render 函数,而 render 函数的作用是什么呢?我们看到在/src/core/instance/lifecycle.js 中有这么一段代码:



意思就是,通过 Watcher 的绑定,每当数据发生变化时,执行_update 的方法,此时会先执行 vm._render(),在这个 vm._render()中,我们的 render 函数会执行,而得到 VNode 对象。



VNode 对象是个什么东西?VNode 就是 Vuejs 2.0 中的 Virtual DOM,在 Vuejs 2.0 中,相较 Vuejs 1.0 引入了 Virtual DOM 的概念,这也是 Vuejs 2.0 性能提升的一大关键。Virtual DOM 有多种实现方式,但基本思路都是一样的,分为两步:

1、Javascript 模拟 DOM 模型树

在 Vuejs 2.0 中 Javascript 模拟 DOM 模型树就是 VNode,render 函数执行后都会返回 VNode 对象,为下一步操作做准备。在/src/core/vdom/vnode.js 中,我们可以看到 VNode 的具体数据结构:



VNode 的数据结构中还有 VNodeData, VNodeDirective, VNodeComponentOptions,这些数据结构都是对 DOM 节点的一些描述,本文不一一介绍。读者可以根据源码来理解这些数据结构。(PS:Vuejs 使用了 flow,标识了参数的静态类型,对理解代码很有帮助^_^)

2、DOM 模型树通过 DOM Diff 算法查找差异,将差异转为真正 DOM 节点

我们知道 render 函数执行生成了 VNode,而 VNode 只是 Virtual DOM,我们还需要通过 DOM Diff 之后,来生成真正的 DOM 节点。在 Vuejs 2.0 中,是通过/src/core/vdom/patch.js 中的 patch(oldVnode, vnode ,hydrating)方法来完成的。


该方法有三个参数 oldVnode 表示旧 VNode,vnode 表示新 VNode,hydrating 表示是否直接使用服务端渲染的 DOM 元素,这个本文不作讨论,在服务端渲染篇再详细介绍。


其主要逻辑为当 VNode 为真实元素或旧的 VNode 和新的 VNode 完全相同时,直接调用 createElm 方法生成真实的 DOM 树,当 VNode 新旧存在差异时,则调用 patchVnode 方法,通过比较新旧 VNode 节点,根据不同的状态对 DOM 做合理的添加、删除、修改 DOM(这里的 Diff 算法有兴趣的读者可以自行阅读 patchVnode 方法,鉴于篇幅不再赘述),再调用 createElm 生成真实的 DOM 树。

Vue 的渲染小结

回过头来看,这里的渲染逻辑并不是特别复杂,核心关键的几步流程还是非常清晰的:


1、new Vue,执行初始化


2、挂载 $mount 方法,通过自定义 render 方法、template、el 等生成 render 函数


3、通过 watcher 监听数据的变化,当数据发生变化时


4、render 函数执行生成 VNode 对象


5、通过 patch 方法,对比新旧 VNode 对象,通过 DOM Diff 算法,添加、修改、删除真正的 DOM 元素


至此,整个 new Vue 的渲染过程完毕。


作者介绍:


王鹤,腾讯前端高级工程师,参与过 QQ 情侣、QQ 星影联盟、QQ 个性化装扮等项目的研发。秉承「不想当产品经理的程序员,不是好的设计师」。除敲敲代码外,对产品、设计、摄影也有一定的兴趣。欢迎一起交流,谈理(CHUI)想(SHUI),聊人(BA)生(GUA)~


本文转载自公众号小时光茶舍(ID:gh_7322a0f167b5)。


原文链接:


https://mp.weixin.qq.com/s/-ykt8u1N50vRnFp2A_4m8Q


2019-08-21 10:381788

评论

发布
暂无评论
发现更多内容

从零开始学习3D可视化之拾取

ThingJS数字孪生引擎

大前端 可视化 3D 3D可视化 数字孪生

深度 | 字节跳动微服务架构体系演进

字节跳动 微服务 云原生 Service Mesh 服务网格 火山引擎

搭建工具提升DDD开发效率

中原银行

领域驱动设计 DDD 中原银行

5月新品速递:EdgeBoard车型识别软硬一体方案,轻松实现智慧车辆管理

百度大脑

5月盘点 上新

HTAP | MySQL 到 ClickHouse 的高速公路

RadonDB

MySQL Clickhouse Xenon

大数据分析与运营(三)

soho

AI未来 - 李开复 - 未来8成的工作受影响 - 读后感-John 易筋 ARTS 打卡 Week 52

John(易筋)

ARTS 打卡计划

架构实战营 模块六:学习总结

👈

架构实战营

通用时区:你应该知道的数据库时区知识

华为云开发者联盟

数据库 时区 GaussDB(DWS) 通用时区 夏令时

Flink Job 概览

Alex🐒

flink 翻译 flink1.13

喜讯 | 拍乐云荣膺「全球云计算大会“云鼎奖”」,先进技术受业界肯定

拍乐云Pano

「免费开源」基于Vue和Quasar的前端SPA项目crudapi后台管理系统实战之文件上传(十)

crudapi

Vue crudapi quasar SPA 文件上传

低碳数字城市和区块链:城市-社区-家庭

CECBC

云原生消息队列RocketMQ:为什么我们选择 RocketMQ

阿里巴巴云原生

「网络安全入门」什么是网络安全

网络安全学海

缓存的世界 Redis(三)

卢卡多多

redis 6月日更

来自小姐姐的灵魂拷问:位运算是什么?

前端森林

计算机网络 位运算 React 二进制

Java高级架构师最新一千道大厂面试真题文字+视频+脑图解析

Java架构追梦

高可用 | Xenon:后 MHA 时代的选择

RadonDB

MySQL 高可用 Xenon

对于后端服务器,框架是怎样的?处理事务的逻辑是怎样的?你了解多少?

奔着腾讯去

c++ io 服务器 信号机制 事件

解Bug之路-ZooKeeper集群拒绝服务

无毁的湖光

Java zookeeper dubbo

通俗易懂的redis发布订阅原理实现!

李阿柯

面试 消息队列 redis cluster

索信达控股首席科学家张磊:人工智能在银行业的应用

索信达控股

大数据 金融科技 银行 银行数字化转型

GitHub 近两万 Star,无需编码,可一键生成前后端代码,这个开源项目有点强!

程序员生活志

和12岁小同志搞创客开发:两个控制器之间如何实现通信?

不脱发的程序猿

DIY 单片机 创客 Arduino

数仓分层架构如何设计?

奔向架构师

数据库 数据仓库 数据架构

[译] R8 优化:类常量操作

Antway

6月日更

问题定位 | XtraBackup 8.0 数据重建避坑事件始末

RadonDB

MySQL Xenon XtraBackup

2021年5月券商App行情刷新及交易体验评测报告

博睿数据

博睿数据 券商App 性能评测

央行约谈!支付宝发布公告:打击虚拟货币交易!

CECBC

区块链技术用在影视行业,能保证编剧们不再被抄袭被欠薪吗?

CECBC

Vuejs 2.0源码解析之渲染篇_语言 & 开发_王鹤_InfoQ精选文章