随着前端框架的快速更新迭代,现在的主流前端框架之一Vue.js迎来了它的新版本3.0。
在今年2020的5月28日,Vue.js的作者尤雨溪公布了Vue3的整个设计过程,让我们来了解一下吧。
简单的来说,这个框架的版本更新一定是为了弥补前一个版本的不足之处,回想一下Vue2的出现也已经有四年之久了,尤大的团队也一直在研究新的版本的Vue,也就是Vue3。
翻阅了一下他们的Vue3设计过程可以了解到版本更新的原因有以下两点:
- 现在的主流浏览器对 JavaScript 语言已经有了非常好的支持,其中包括了后来ES6更新的语法,而Vue2使用的大多都是ES6之前的语法,所以现在可以通过利用新的语法对Vue框架进行更新,以提高性能
- Vue2代码库中的设计和体系架构存在一定的问题,所以不得不对其进行更正
那么Vue3到底更新了什么呢?它有哪些新的特性?它相对于Vue2的优势是什么?Vue3如何进行使用?
别急,本文就将带着大家提前体验一下Vue3,以上的问题也会一一得到解答,大家也可以从中感受一下Vue3的魅力。
一、Vue3的亮点
在体验Vue3之前,我们先来了解一下Vu3到底有哪些亮点之处
总共有6大点:
- Performance(性能比vue2的runtime快2倍左右)
- Tree shaking support(按需打包模块)
- Better TypeScript support(更好的TS代码支持)
- Composition API(组合API)
- Custom Renderer API(自定义渲染API)
- Fragment, Teleport, Suspense
(1)Performance
在原本的Vue2中,每次DOM某个地方需要更新都需要遍历整个虚拟DOM,然后判断哪个地方需要进行更新,然后再重新渲染。而Vue3选择了一种新的渲染策略,即采用某些方式使得每次需要更新DOM时,无需遍历整个虚拟DOM树就可以检测到改变而进行DOM更新。
(2)Tree shaking support
这个就是为了实现按需打包,我们知道Vue框架里有很多的API和模块,但是一个项目中不可能用到所有的API和模块,所以该功能就可以实现在打包时去除掉Vue中没有被用到的模块和API,使得整个项目大小大幅度减小。
根据官方的说法,Vue3如果只写入了一个 Hello World
,即没有用到任何的模块API,打包后的大小大约为 13.5kb
;若用到了所有的Vue3所有的模块,打包后的大小也就只有大约 22.5kb
;而Vue2若只写入了 Hello World
,并没有用到任何的模块API,打包后的大小大约为 32kb
左右,这就是因为Vue2中没有 Tree shaking
的原因。
从中可以看出,Vue3比Vue2更轻量。
(3)Better TypeScript support
为了更好的用户使用,Vue3使用了TS重新对Vue框架进行了重写,增加了对TS更好的支持。
(4) Composition API
Composition API
是为了解决原本Vue项目中代码逻辑分散而导致不易维护的问题
首先先来看一个图
这幅图里颜色相同的区域表示的是实现同一个功能的代码区域
图中左侧的是Vue2的一个API,叫做 Options
,就是我们代码中的 data
、methods
、mounted
……这些存储数据或方法的对象
我们都知道,在大的项目中,我们要实现一个组件的完整功能,需要上百甚至上千行代码。例如,我们要做一个轮播图组件,现在把图中左侧这一列图像看成是轮播图组件的代码,假设这里有1000行代码。
首先我们在 methods
里写了实现轮播功能的方法,然后在 mounted
里写上在进入页面时自动开启轮播功能的代码,此时又想起来应该加一个功能,在鼠标移入该组件时应暂停轮播,所以又在 methods
里新增了一个方法,如图
这时你就会有这种感受,仅仅为了实现一个轮播的功能,却在这个页面中的不同地方写了不同的代码,那这样维护起来就特别的麻烦了,假设你的轮播功能出bug了,你需要排查问题,这时你先检查 实现轮播功能的代码
有无问题,然后又往下翻几百行代码找到 鼠标移入暂停轮播的代码
,检查一下是否有问题,再又往下翻几百行代码找到 进入页面开启轮播的代码
是否又问题。最终一顿分析才能查出bug。
这样一来,代码逻辑在页面中比较分散,很难维护,所以Vue3舍弃了这种 Options API
,而换用 Composition API
,也就是图中右侧的部分。
我们来看一下,刚才举的例子如果换用 Composition API
会是什么样
所有实现轮播这一功能的代码都放在一起,也就自然方便了维护
(5) Custom Renderer API
Custom Renderer API
也算是Vue3一个很不错的亮点了,它的作用就是将我们的代码绘制到 canvas
画布上
(6) Fragment
Vue3实现了不再限于模板中的单个根节点
在Vue2中大家应该遇到过这样的情况
这个报错的原因是因为,Vue2要求组件模板中只能有一个根节点,而不可以有多个根节点,如图
图中这种情况就是有两个根节点,所以会报错。所以我们可以通过给这两个标签外部加一个节点来解决报错问题,如图
这样就不会在报错了,但有时会觉得这样特别麻烦,所以Vue3解决这个报错问题,实现了模板中不限于单个根节点,即即使有多个根节点也不会报错
二、Vue3的性能
在Vue3中,重写了虚拟dom的实现,并且针对编译模板进行了优化。
在上面我们讲到了,在Vue2中,每次更新dom都需要重新遍历整个虚拟dom树,检测到变化点再去做相应的更新。但是Vue3在每次更新dom时就不需要遍历整个虚拟dom树了,这是为什么呢?
这里先放上两个网址,分别可以查看到Vue2和Vue3的模板编译是如何的:
- Vue2-template:https://template-explorer.vuejs.org/#%3Cdiv%20id%3D%22app%22%3E%0A%20%20%7B%7B%20msg%20%7D%7D%0A%3C%2Fdiv%3E
Vue的渲染过程是先将html模板生成一个 render
函数,然后再根据该 render
函数进行渲染的,我们首先来看一下Vue2的模板编译情况
图中左侧部分就是我们平时写的html模板,右侧部分则是该html模板编译后生成的 render
函数
接下来我们再来看一下Vue3的模板编译情况是如何的
很明显的可以看到,对比Vue2的 render
函数,在图中的第7行代码中多出了一个数字 1
和 /* TEXT */
,这是针对html模板打上了一个标记,告诉程序这个位置的数据是动态的(html模板中的{{ msg }}
是可能会改变的,是一个动态的数据),所以之后如果要更新dom,就会针对打上标记的位置进行遍历检测。
现在我们再在html模板上写上一个静态的数据,看看它是如何编译模板的
我们可以看到,我先给最外部的 div
加上了一个动态的 class
属性,即:class="isShow"
,所以编译后的 render
函数中的第11行代码中,有一个数字 9
、 注释 /* TEXT, PROPS */
以及 ["class"]
,表示html模板该处的 text
文本部分是动态的,属性 class
也是动态的,所以在后面更新dom时,会对该处的这两个值进行遍历检测;
另外我们给属性name
赋值了 box
,该值是一个静态固定的值,所以在图中右侧的 render
函数中我们并没有看到对其有任何的标记,所以在之后更新dom时,会自动忽略该处的值。
这就是Vue3通过标记来追踪动态数据的绑定,从而节约遍历虚拟dom的大部分开销。
结合官方的说法,Vue3比Vue2的页面渲染速度提升了1.3倍~2倍,SSR(服务器渲染)速度提高了2倍~3倍
这里放上两者性能的对比图