Vue 3.0到底怎么变快?🚀

简介: Vue 3.0到底怎么变快?🚀

前言🖖


自从Vue3.0发布后一直想要学习一下新特性,谁曾想到一直鸽到了现在。面对前端行业不学习就被淘汰的内卷环境什么都要学起来呀,学的时候索性记录总结一下。


Vue 3.0亮点🤩


  • Vue 3.0性能比Vue 2.x1.2~2
  • Tree-Shaking:按需编译,体积比Vue 2.x
  • 新推出的Composition API 使组件更易维护
  • 更好TypeScript支持
  • Custom Render API:暴露了自定义渲染API
  • 更多先进牛逼的组件

总的来说就是更快更小更舒服.


Vue 3.0到底是如何变快🤔


大家都说Vue 3.0变快了,那他到底是哪里快了呢?做了哪些优化才会让它有这种效果?


diff算法优化🦄

  • 熟悉Vue 2.x的兄弟都知道渲染真实DOM的开销是很大的,比如我们修改某个数据直接渲染在真实DOM上会引起DOM树的重排重绘。
  • 于是就有了diff算法可以根据真实DOM生成一棵虚拟DOM树(Virtual DOM)从而计算出 Virtual DOM 中被改变的部分,然后针对该部分进行原生DOM操作,而不用重新渲染整个页面。

image.png

  • 值得注意的是diff算法是对树的每一层进行遍历从而找出不同进行替换(图片来自网络),而Vue3.0diff算法就在此做出了优化。
  • 举个例子说明一下吧,在下面有两个p标签一个是不会变的数据一个是绑定的数据,当sth改变的时候,Vue会生成新的虚拟DOM然后和旧的进行对比。
<div>
    <p>hi 小卢!!</p>
    <p>{{sth}}</p>
  </div>
复制代码

Vue 2.x

image.png

Vue 3.0

image.png

  • 从上面的图例可知在Vue 2.xdiff算法会将两棵DOM树的所有节点进行对比,但实际变化的只有双向绑定的那个。
  • Vue 3.0中会在创建虚拟DOM的时候将会变化的内容进行静态标记,这样diff算法的时候直接对比有标记的,对比的少了自然而然速度就变快了。

Vue 3 Template Explorer中显示如下image.png上面flag的1是Patchflag枚举,取值为1代表这个元素的文本是动态绑定的

附录:PatchFlags
export const enum PatchFlags {Ⅰ
    TEXT = 1,1/动态文本节点
    CLASS = 1<<1,1/ 2// 动态 classSTYLE= 1<<2,// 4//动态 style
    PROPS = 1<< 3,// 8// 动态属性,但不包含类名和样式
    FULL_PROPS = 1<<4,// 16 //具有动态 key属性,当key改变时,需要进行完整的 diff 比较。
    HYDRATE_EVENTS = 1<<5,// 32//带有监听事件的节点
    STABLE_FRAGMENT = 1<<6,// 64//一个不会改变子节点顺序的 fragment
    KEYED_FRAGMENT = 1<<7,// 128//带有key属性的 fragment 或部分子字节有
    keyUNKEYED_FRAGMENT = 1<<8,// 256//子节点没有key 的 fragment
    NEED_ PATCH =1<<9,//512//一个节点只会进行非 props比较
    DYNAMIC_SLOTS = 1 << 10,//1024 // 动态的插槽
    // SPECIAL FLAGS (下面是特殊的)--------------------------------------------------------- 
    // 以下是特殊的flag,不会在优化中被用到,是内置的特殊flag
    // 表示他是静态节点,他的内容永远不会改变,对于hydrate的过程中,不会需要再对其子节点进行diff
    HOISTED = -1,
    BAIL = -2, // 用来表示一个节点的diff应该结束
}
复制代码

如果想更多了解PatchFlags可以阅读Vue3的patchFlags超详细讲解


hoistStatic(静态提升)🦓

  • 这就好比你每天去便利店买咖啡,每次买的时候店员都问你需要什么,当时间久了店员熟悉了你的习惯就省去了店员问你'要什么'的这个过程,直接给你所需要的商品。
  • 顾名思义静态提升就是把不参与更新的静态元素给提升出来,说白了就是类似把变量变成常量不进行重新创建
  • Vue 2.x中无论元素是否参与更新每次都会重新创建,然后渲染,这对于性能肯定是会有些许损耗。

举个例子说明一下

<div>
  <div>Hello Vue3.0</div>
  <div>hi 小卢!</div>
  <p>{{sth}}</p>
</div>
复制代码

未使用静态提升

image.png使用静态提升

image.png

  • 从上面的两幅对比图可以明显的看到未使用静态提升的变量会放在render函数里面,每次更新数据都会重新创建并渲染出来
  • 使用了静态提升的元素会被提取出来放在render函数外面,这样下次更新的时候就不会把这个元素重新创建,创建的元素少了,自然而然速度也就起来了。
  • Vue 3.0中就是启用了静态提升把不需要更新的元素放到外面只创建一次,在渲染的时候直接复用即可。

cacheHandlers(事件侦听器缓存)🐲

  • 从上面的diff算法优化中我们知道,根据PatchFlags会把动态绑定的元素加上静态标记,那一个事件函数自然就会被加上静态标记,在diff中会将有标记的元素进行对比来更新数据。
  • 如果我们用的这个函数从头到尾都没有改变过,那是不是也可以让它像静态元素一样不进行对比呢?
  • 那我都这么问了答案肯定是:可以的!
  • 顾名思义事件侦听器缓存就是把没有改变过的事件给侦听到了,然后缓存。
  • 就好比刚学Vue的你不懂各种api和优秀的组件,每次用到的时候都会去查看看怎么做,但是当你做久了之后自然你就熟能生巧了,没有得到消息说api更新了,自然就不会去查找文档对比有什么差异。

举个例子说明一下

<div>
  <div>Hello Vue3.0</div>
  <div>hi 小卢!</div>
  <p>{{sth}}</p>
  <div @click="doSthing">我是一个点击事件</div>
</div>
复制代码

未使用事件侦听器缓存

image.png使用事件侦听器缓存

image.png

  • 上面转换过的代码如果看不懂的话没有关系,我们只需要观察它是否有静态标记即可。
  • 从上面两幅对比图可以看出来在使用了事件监听器缓存的情况下它的静态标记8消失了,就说明在Vue 3.0在每次数据更新时不会对它进行对比了,对比的少了自然而然速度也就加快


写在最后👌


  • 以上就是Vue 3.0变快做的一些优化。
  • 总的来说Vue 3.0通过优化diff算法让该对比的进行对比,不该对比的不进行对比。
  • 通过静态提升事件侦听缓存来减少不变的元素创建的频率,加快了运行速度。
  • 如果您觉得这篇文章有帮助到您的的话不妨点赞支持一下哟~~😛


参考👈



相关文章
|
2月前
|
缓存 JavaScript UED
Vue3中v-model在处理自定义组件双向数据绑定时有哪些注意事项?
在使用`v-model`处理自定义组件双向数据绑定时,要仔细考虑各种因素,确保数据的准确传递和更新,同时提供良好的用户体验和代码可维护性。通过合理的设计和注意事项的遵循,能够更好地发挥`v-model`的优势,实现高效的双向数据绑定效果。
139 64
|
2月前
|
JavaScript 前端开发 API
Vue 3 中 v-model 与 Vue 2 中 v-model 的区别是什么?
总的来说,Vue 3 中的 `v-model` 在灵活性、与组合式 API 的结合、对自定义组件的支持等方面都有了明显的提升和改进,使其更适应现代前端开发的需求和趋势。但需要注意的是,在迁移过程中可能需要对一些代码进行调整和适配。
114 60
|
8天前
|
JavaScript API 数据处理
vue3使用pinia中的actions,需要调用接口的话
通过上述步骤,您可以在Vue 3中使用Pinia和actions来管理状态并调用API接口。Pinia的简洁设计使得状态管理和异步操作更加直观和易于维护。无论是安装配置、创建Store还是在组件中使用Store,都能轻松实现高效的状态管理和数据处理。
31 3
|
2月前
|
前端开发 JavaScript 测试技术
Vue3中v-model在处理自定义组件双向数据绑定时,如何避免循环引用?
Web 组件化是一种有效的开发方法,可以提高项目的质量、效率和可维护性。在实际项目中,要结合项目的具体情况,合理应用 Web 组件化的理念和技术,实现项目的成功实施和交付。通过不断地探索和实践,将 Web 组件化的优势充分发挥出来,为前端开发领域的发展做出贡献。
38 8
|
2月前
|
存储 JavaScript 数据管理
除了provide/inject,Vue3中还有哪些方式可以避免v-model的循环引用?
需要注意的是,在实际开发中,应根据具体的项目需求和组件结构来选择合适的方式来避免`v-model`的循环引用。同时,要综合考虑代码的可读性、可维护性和性能等因素,以确保系统的稳定和高效运行。
32 1
|
2月前
|
JavaScript
Vue3中使用provide/inject来避免v-model的循环引用
`provide`和`inject`是 Vue 3 中非常有用的特性,在处理一些复杂的组件间通信问题时,可以提供一种灵活的解决方案。通过合理使用它们,可以帮助我们更好地避免`v-model`的循环引用问题,提高代码的质量和可维护性。
41 1
|
2月前
|
JavaScript
在 Vue 3 中,如何使用 v-model 来处理自定义组件的双向数据绑定?
需要注意的是,在实际开发中,根据具体的业务需求和组件设计,可能需要对上述步骤进行适当的调整和优化,以确保双向数据绑定的正确性和稳定性。同时,深入理解 Vue 3 的响应式机制和组件通信原理,将有助于更好地运用 `v-model` 实现自定义组件的双向数据绑定。
|
2月前
|
存储 JavaScript 前端开发
vue3的脚手架模板你真的了解吗?里面有很多值得我们学习的地方!
【10月更文挑战第21天】 vue3的脚手架模板你真的了解吗?里面有很多值得我们学习的地方!
vue3的脚手架模板你真的了解吗?里面有很多值得我们学习的地方!
|
2月前
|
JavaScript 索引
Vue 3.x 版本中双向数据绑定的底层实现有哪些变化
从Vue 2.x的`Object.defineProperty`到Vue 3.x的`Proxy`,实现了更高效的数据劫持与响应式处理。`Proxy`不仅能够代理整个对象,动态响应属性的增删,还优化了嵌套对象的处理和依赖追踪,减少了不必要的视图更新,提升了性能。同时,Vue 3.x对数组的响应式处理也更加灵活,简化了开发流程。
|
2月前
|
JavaScript 前端开发 开发者
Vue 3中的Proxy
【10月更文挑战第23天】Vue 3中的`Proxy`为响应式系统带来了更强大、更灵活的功能,解决了Vue 2中响应式系统的一些局限性,同时在性能方面也有一定的提升,为开发者提供了更好的开发体验和性能保障。
78 7

热门文章

最新文章