【Vue原理解析】之异步与优化
引言
Vue是一款流行的JavaScript框架,它提供了一些强大的特性来提升应用程序的性能和用户体验。在本文中,我们将深入探讨Vue的异步更新机制和一些优化技巧,帮助您更好地理解和应用这些特性。
异步更新机制
Vue使用异步更新机制来提高渲染性能。当数据发生变化时,Vue并不立即重新渲染整个组件树,而是将更新操作推入一个队列中,并在下一个事件循环中执行。这样可以将多个数据变化合并为一个更新操作,减少不必要的重复渲染。
nextTick方法
Vue提供了nextTick
方法来处理异步更新。它接受一个回调函数作为参数,在下次DOM更新循环结束后执行该回调函数。这样可以确保在DOM更新完成后再进行一些操作。
<template><div><p>{{ message }}</p><button @click="updateMessage">Update Message</button></div>
</template><script>
export default {data() {return {message: "Hello, Vue!",}},methods: {updateMessage() {this.message = "Updated Message"this.$nextTick(() => {console.log("DOM updated")})},},
}
</script>
$forceUpdate
合理使用$forceUpdate
方法来强制组件重新渲染,尤其在某些特殊情况下需要手动触发组件更新时。
<template><div><p>{{ message }}</p><button @click="updateMessage">Update Message</button></div>
</template>
<script>
export default {data() {return {message: "Hello, Vue!",}},methods: {updateMessage() {// 手动修改DOM元素的内容document.querySelector("p").textContent = "Updated Message"// 强制组件重新渲染this.$forceUpdate()},},
}
</script>
在上述代码中,我们定义了一个包含一个按钮的Vue组件。当点击按钮时,会手动修改DOM元素的内容,并通过调用$forceUpdate
方法强制组件重新渲染。这样可以确保即使数据没有发生变化,也能强制刷新组件以更新视图。 需要注意的是,在大多数情况下,Vue会自动追踪数据变化并进行相应的更新,不需要手动触发组件更新。只有在特殊情况下(如直接修改DOM元素),才需要使用$forceUpdate
方法。 然而,应该谨慎使用$forceUpdate
方法,因为它会跳过Vue的优化机制,并可能导致性能下降。只有在确实需要手动触发组件更新时,才应该使用$forceUpdate
方法。
$set
<template><div><ul><li v-for="item in items" :key="item.id">{{ item.name }}</li></ul><button @click="addItem">Add Item</button></div>
</template><script>
export default {data() {return {items: [],}},methods: {addItem() {const newItem = { id: Date.now(), name: "New Item" }this.$set(this.items, this.items.length, newItem)},},
}
</script>
在上述代码中,我们定义了一个包含一个按钮的Vue组件。当点击按钮时,会向items数组中添加一个新的项。通过使用this.$set
方法,我们可以确保新添加的项是响应式的,并能够触发视图更新。
优化技巧
除了异步更新机制,Vue还提供了一些优化技巧来进一步提升应用程序的性能和用户体验。
列表渲染优化
在列表渲染时,为每个列表项添加唯一的key
属性可以帮助Vue更高效地更新DOM。Vue会根据key
属性来判断哪些列表项需要更新,哪些需要新增或删除。
<template><ul><li v-for="item in items" :key="item.id">{{ item.name }}</li></ul>
</template><script>
export default {data() {return {items: [{ id: 1, name: "Item 1" },{ id: 2, name: "Item 2" },{ id: 3, name: "Item 3" },],}},
}
</script>
计算属性和侦听器
使用计算属性可以缓存计算结果,避免重复计算。而侦听器可以监听数据的变化,并在变化时执行相应的操作,避免不必要的计算。
<template><div><p>Width: {{ width }}</p><p>Height: {{ height }}</p><p>Area: {{ area }}</p></div>
</template><script>
export default {data() {return {width: 10,height: 5,}},computed: {area() {return this.width * this.height},},watch: {width(newWidth) {console.log("Width changed:", newWidth)},height(newHeight) {console.log("Height changed:", newHeight)},},
}
</script>
合理使用keep-alive
使用keep-alive
组件可以缓存组件的状态,避免重复渲染和销毁。特别适用于包含表单输入、列表等需要保留状态的场景。
<template><div><keep-alive><component :is="currentComponent"></component></keep-alive><button @click="toggleComponent">Toggle Component</button></div>
</template><script>
import ComponentA from "./ComponentA.vue"
import ComponentB from "./ComponentB.vue"export default {data() {return {currentComponent: "ComponentA",}},components: {ComponentA,ComponentB,},methods: {toggleComponent() {this.currentComponent =this.currentComponent === "ComponentA" ? "ComponentB" : "ComponentA"},},
}
</script>
懒加载(Lazy Loading)
合理使用懒加载(Lazy Loading)来延迟加载组件或资源,减少初始加载时间
// vue2
const MyComponent = () => import('./MyComponent.vue')// vue3
import { defineAsyncComponent } from 'vue'const AsyncComponent = defineAsyncComponent(() => import('./AsyncComponent.vue'))
在上述代码中,我们使用defineAsyncComponent
函数来定义异步组件。该函数接受一个返回import()
函数的回调作为参数,用于动态导入组件文件。这样,在需要使用AsyncComponent
组件时才会进行实际的加载。
与Vue 2不同,Vue 3中的异步组件不再需要通过动态导入返回一个Promise对象。而是直接通过defineAsyncComponent
函数来定义异步组件。 需要注意的是,在Vue 3中,异步组件默认会自动进行Suspense处理。可以在父级组件中使用<Suspense>
包裹异步组件,并提供一个fallback内容作为加载过程中显示的占位符。
<template><div><Suspense><template #default><AsyncComponent /></template><template #fallback><div>Loading...</div></template></Suspense></div>
</template>
函数式组件(Functional Components)
当使用Vue的函数式组件时,可以通过functional
选项来定义一个函数式组件。下面是一个示例,展示了如何使用Vue的函数式组件:
<template functional><div><p>{{ props.message }}</p><button @click="props.onClick">Click me</button></div>
</template>
在上述代码中,我们使用<template functional>
来定义一个函数式组件。在函数式组件中,我们可以通过props
对象来访问传递给组件的属性。这样可以避免创建响应式数据和实例状态。 需要注意的是,在函数式组件中无法使用data
、computed
、methods
等选项。如果需要计算属性或方法,可以通过传递额外的参数来实现。
<template functional><div><p>{{ computeMessage(props.message) }}</p></div>
</template>
<script>
export default {methods: {computeMessage(message) {return message.toUpperCase()},},
}
</script>
在上述代码中,我们通过传递额外的参数来调用计算属性computeMessage
。 通过合理使用函数式组件,我们可以减少不必要的实例化和响应式开销,并提升应用程序的性能。特别适用于那些没有状态或只依赖传入属性的简单组件。 需要注意的是,函数式组件不支持在模板中使用自定义指令和过滤器,并且无法访问Vue实例上的方法和属性。
另:使用虚拟滚动(Virtual Scrolling)或分页加载等技术来处理大量数据列表,避免一次性渲染大量DOM元素。
注意事项
- 避免频繁地使用
$forceUpdate
方法,因为它会跳过Vue的优化机制,可能导致性能下降。 - 当使用异步更新机制时,需要注意避免对异步更新的数据进行同步操作,以免引起意外的结果。
- 在使用
v-for
渲染大量列表时,尽量避免在每个列表项中使用复杂的计算属性或方法,以减少不必要的计算开销。
总结
在本文中,我们深入探讨了Vue的异步更新机制和一些优化技巧。异步更新机制通过将多个数据变化合并为一个更新操作,提高了渲染性能。而优化技巧如列表渲染优化、计算属性和侦听器、合理使用keep-alive
等,进一步提升了应用程序的性能和用户体验。
通过合理应用这些特性和技巧,您可以构建出更高效、更流畅的Vue应用程序。
相关文章:

【Vue原理解析】之异步与优化
引言 Vue是一款流行的JavaScript框架,它提供了一些强大的特性来提升应用程序的性能和用户体验。在本文中,我们将深入探讨Vue的异步更新机制和一些优化技巧,帮助您更好地理解和应用这些特性。 异步更新机制 Vue使用异步更新机制来提高渲染性…...

mybatis、mysql 创建时间(create_time)异常自动更新为当前时间
目录标题 一、问题二、原因三、解决 一、问题 bug: mybatis更新代码没有修改时间,但是时间会自动更新为当前时间。 。。。 被坑了挺久 二、原因 可能是创建表的时候, Navicat Premium 等可视化工具给你整活了。。。 三、解决 取消勾选。 注意&…...

shardingsphere 加载慢 优化
shardingsphere加载慢 优化 原因: 启动速度变慢(元数据扫描耗时较长) 占用内存增多(大量单表和元数据对象) 那是因为默认扫描1张表.一张一张加载巨慢,添加以下配置增加到20张表同时扫描 错误: 如果你数据库最大连接不到20就会报错,请按照数据…...

我这些年对于自动化测试的理解
📢专注于分享软件测试干货内容,欢迎点赞 👍 收藏 ⭐留言 📝 如有错误敬请指正!📢交流讨论:欢迎加入我们一起学习!📢资源分享:耗时200小时精选的「软件测试」资…...

Java安全架构 JCA、JCE、JSSE、JAAS
Java语言拥有三大特征:平台无关性、网络移动性和安全性,而Java安全体系结构对这三大特征提供了强大的支持和保证, Java安全体系结构总共分为4个部分: (1)JCA( Java Cryptography Architecture…...

面试经典(4/150)删除有序数组中的重复项 II
面试经典(4/150)删除有序数组中的重复项 II 给你一个有序数组 nums ,请你 原地 删除重复出现的元素,使得出现次数超过两次的元素只出现两次 , 返回删除后数组的新长度。不要使用额外的数组空间,你必须在 原…...

使用WildCard充值ChatGPT Plus 会员
登录 wildCard官网 基于国内手机号注册账号,使用支付宝验证付款即可自动申请国外银行卡,WildCard的开卡费是9.9美元, 没有后续的月费用.订阅chatgpt plus会员服务的操作图文指南见链接 chatgpt plus会员订阅指南...

element-plus使用el-date-picker组件时,如何禁止用户选择当前时间之后的日时分秒
element-plus使用el-date-picker组件时,如何禁止用户选择当前时间之后的日时分秒 例: 当前时间为2023-11-15 14.24,不能选择这之后的时分秒。(禁止用户选择2023-11-15 14.28) <el-date-pickerv-model"form.s…...

keepalived安装配置(服务器主备、负载均衡)
系统拓扑 安装keepalived 主备服务器上都需要安装 在线安装 yum install -y keepalived 离线安装 # todo 服务器准备 虚拟机ip:192.168.11.56 主服务器:192.168.11.53 备服务器:192.168.11.54 配置文件修改 keepalived安装之后&…...

盘点一款制作电子杂志的网站,小白也能快速上手
电子杂志作为一种时尚、环保、便捷的宣传形式,越来越受到各行各业的青睐。无论是企业宣传、产品推广,还是个人分享,电子杂志都能展现出独特的魅力。而制作电子杂志,不再是专业人士的专属,现在小白也能快速上手&#…...

全域全自主建设,亚信科技AntDB数据库助力广电5G业务上线运行
自2019年6月,中国广电成功获得5G牌照以来,迅速推进网络建设目标,成为5G网络覆盖广、应用场景多、用户体验出色的第四大运营商。其依托全球独有的700MHz频谱资源,具备覆盖能力强、容量足、速率高的优势。通过不断深化和中国移动的共…...

使用 SSH 密钥进行身份验证
使用 SSH 密钥进行身份验证可以提高安全性,并免去每次登录时输入密码的麻烦。以下是使用 SSH 密钥进行身份验证的步骤: 生成密钥对:在本地计算机上生成 SSH 密钥对。打开终端并执行以下命令: ssh-keygen -t rsa -b 4096这将生成…...

国内最受欢迎的电商API接口调用京东商品详情数据
国内实用的API接口 国内最受欢迎的7大API供应平台对比和介绍 本文将介绍7款API供应平台:电商API数据、百度e、Apix、数说聚合、通联数据、HaoService、datasift 。排名不分先后! 免费实用的API接口 第一部分 1、电商数据(API数据接口_开发者…...

windows远程桌面登录ubuntu,黑屏闪退,
首先需要安装xrdp以后才能远程登录,安装命令 sudo apt-get install xrdp 这里需要注意一个问题,如果在ubuntu 本地登录的情况下,windows远程登录会出现黑屏甚至闪退的问题。原因是本地登录和远程登陆是互斥的,本地登录了就不能远…...

12-使用vue2实现todolist待办事项
个人名片: 😊作者简介:一名大二在校生 🤡 个人主页:坠入暮云间x 🐼座右铭:懒惰受到的惩罚不仅仅是自己的失败,还有别人的成功。 🎅**学习目标: 坚持每一次的学习打卡 文章…...

微信小程序授权登录?
wxml <!-- 示例:在wxml中创建一个授权登录按钮 --> <button bindtap"getUserInfo">授权登录</button> js // 用户点击授权登录按钮时触发的事件处理函数getUserInfo: function (e) {wx.getUserProfile({desc: 用于完善会员资料, // 授…...

React 18 + Hooks +Ts 开发中遇到的问题及解决方案!
这篇文章是用来专门记录关于React 18 Hooks Ts 开发中遇到的问题及解决方案 Q1 问题描述: TS7016: Could not find a declaration file for module js-export-excel. /Users/zhangliangliang/WebstormProjects/daizhang-system-front/node_modules/js-export-exc…...

推荐一个非常好用的uniapp的组件库【TMUI3.0】
文章目录 前言官网地址如何使用?注意事项后言 前言 hello world欢迎来到前端的新世界 😜当前文章系列专栏:前端系列文章 🐱👓博主在前端领域还有很多知识和技术需要掌握,正在不断努力填补技术短板。(如果…...

LeetCode(19)最后一个单词的长度【数组/字符串】【简单】
目录 1.题目2.答案3.提交结果截图 链接: 58. 最后一个单词的长度 1.题目 给你一个字符串 s,由若干单词组成,单词前后用一些空格字符隔开。返回字符串中 最后一个 单词的长度。 单词 是指仅由字母组成、不包含任何空格字符的最大子字符串。 …...

使用Docker本地安装部署Drawio绘图工具并实现公网访问
目录 前言 1. 使用Docker本地部署Drawio 2. 安装cpolar内网穿透工具 3. 配置Draw.io公网访问地址 4. 公网远程访问Draw.io 前言 提到流程图,大家第一时间可能会想到Visio,不可否认,VIsio确实是功能强大,但是软件为收费&…...

IDEA导入jar包
通过maven导入本地包 mvn install:install-file -DfileD:\WebProject\ERP\zhixing-heyue-erp-server\zxhy-service-api\src\main\java\com\zxhy\service\api\invoice\baiwang\lib\com_baiwang_bop_sdk_outer_3_4_393.jar -DgroupIdcom.baiwang -DartifactIdbaiwang.open -Dver…...

使用flutter的Scaffold脚手架开发一个最简单的带tabbar的app模板
flutter自带的scaffold脚手架可以说还是挺好用的,集成了appBar,还有左侧抽屉,还有底部tabbar,可以说拿来就可以用了啊,所以我今天也体验了一下,做了一个最简单的demo,就当是学习记录了。 效果展…...

aws服务器配置密码登陆
在 AWS 上,EC2 实例默认使用密钥对进行身份验证,而不是密码登录。不过,你可以通过以下步骤在 EC2 实例上启用密码登录: 登录 AWS 管理控制台并导航到 EC2 服务。 选择要配置密码登录的目标 EC2 实例。 在底部的 “描述” 标签页…...

【命令行魔法:掌握Linux基础工具开发的独门技艺】
本节目标 1.Linux 软件包管理器 2.Linux开发工具 3.Linux编译器-gcc/g使用 4.Linux项目自动化构建工具-make/Makefile 5.Linux第一个小程序-进度条 1.Linux 软件包管理器 yum 什么是软件包 在Linux下安装软件, 一个通常的办法是下载到程序的源代码, 并进行编译…...

虚拟DOM的原理和理解
Virtual DOM前言 在传统的Web开发中,直接操作真实的DOM通常是一个昂贵且低效的操作。为了解决这个问题,Virtual DOM(虚拟DOM)被引入为一个中间层,允许开发者在内存中进行操作,从而避免频繁且不必要的真实DO…...

C# WPF Threads 和 Dispatchers 有什么区别
在C# WPF(Windows Presentation Foundation)中,Threads(线程)和Dispatchers(调度器)之间的关系非常重要,因为WPF是一个基于STA(单线程单元)的UI框架。 Threa…...

【文末送书——数学经典著作】工科必备的数学思维培养
欢迎关注博主 Mindtechnist 或加入【智能科技社区】一起学习和分享Linux、C、C、Python、Matlab,机器人运动控制、多机器人协作,智能优化算法,滤波估计、多传感器信息融合,机器学习,人工智能等相关领域的知识和技术。关…...

【云备份项目两万字总结】服务端篇 -----附源码
项目总结 整体回顾逐步实现utill.hppconfig.hppdata.hpphot.hppservice.hpp 代码 整体回顾 服务端的目标是: 对客户端的请求进行处理管理客户端上传的文件 于客户端进行数据交换,我们需要引入网络,所以我们引入第三方库----httplib.h库&am…...

蓝眼开源云盘部署全过程(手动安装)
环境概述: 系统-Centos7.4 数据库-MySQL8 云盘系统-Tank4.0.1 前提:操作系统已完成安装,有外部网络。 一.安装数据库 cd到合适的目录进行下载安装操作,期间不要切换出去。 wget https://dev.mysql.com/get/mysql80-community-r…...

aliyun Rest ful api V3版本身份验证构造
aliyun Rest ful api V3版本身份验证构造 参考官网:https://help.aliyun.com/zh/sdk/product-overview/v3-request-structure-and-signature?spma2c4g.11186623.0.0.787951e7lHcjZb 构造代码 :使用GET请求进行构造,算法使用sha256 使用postm…...