大家好,我是小杨,一个写了6年前端的老码农。今天咱们来聊聊Vue里的$nextTick
和JavaScript的setTimeout
,这两个都能让代码"等会儿再执行",但到底有啥不同?相信看完这篇你就彻底明白了!
一、先说说$nextTick是干啥的
$nextTick
是Vue提供的一个方法,它的作用简单来说就是:"等Vue完成DOM更新后,再执行我的代码"。
举个例子,我写了个按钮,点击后修改数据,然后立刻获取DOM内容:
html
<template>
<div>
<button @click="updateData">更新数据</button>
<div ref="content">{{ message }}</div>
</div>
</template>
<script>
export default {
data() {
return {
message: '初始消息'
}
},
methods: {
updateData() {
this.message = '新消息'
console.log(this.$refs.content.textContent) // 这里会输出什么?
}
}
}
</script>
猜猜控制台会输出什么?是"新消息"吗?错!实际上会输出"初始消息"!因为Vue的DOM更新是异步的,数据改变后不会立即更新DOM。
这时候就该$nextTick
出场了:
javascript
updateData() {
this.message = '新消息'
this.$nextTick(() => {
console.log(this.$refs.content.textContent) // 现在输出"新消息"了!
})
}
二、定时器setTimeout也能延迟执行
那有人要问了,我用setTimeout
不也一样吗?
javascript
updateData() {
this.message = '新消息'
setTimeout(() => {
console.log(this.$refs.content.textContent) // 也能输出"新消息"
}, 0)
}
确实,在这个简单例子里效果看起来一样。但它们的本质完全不同!
三、核心区别:执行时机不同
特性 | $nextTick | setTimeout |
---|---|---|
触发时机 | Vue的DOM更新周期后立即执行 | 在JS事件循环的定时器阶段执行 |
执行顺序 | 优先于setTimeout | 在$nextTick之后执行 |
用途 | 确保在DOM更新后操作DOM | 通用的延迟执行方法 |
可靠性 | 保证在下次DOM更新循环后执行 | 不保证与DOM更新时机的关系 |
来看个更明显的例子:
javascript
this.message = '更新1'
this.$nextTick(() => {
console.log('nextTick回调')
})
setTimeout(() => {
console.log('setTimeout回调')
}, 0)
this.message = '更新2'
输出顺序永远是:
- nextTick回调
- setTimeout回调
四、什么时候用$nextTick?
-
需要在DOM更新后操作DOM时
- 比如修改数据后获取DOM尺寸
- 动态渲染后初始化第三方库
-
解决Vue的异步更新导致的奇怪问题
- 比如弹窗出现后自动聚焦输入框
javascript
showModal() {
this.isShow = true
this.$nextTick(() => {
this.$refs.input.focus() // 确保modal渲染完成后再focus
})
}
五、什么时候用setTimeout?
-
需要纯粹的时间延迟时
- 比如展示loading后延迟请求
- 动画的延时效果
-
与Vue无关的普通JS延迟逻辑
javascript
fetchData() {
this.loading = true
setTimeout(() => {
// 模拟网络请求
this.data = getData()
this.loading = false
}, 500)
}
六、性能考量
$nextTick
比setTimeout
更高效,因为:
- 它利用了Vue自身的更新机制
- 避免了浏览器不必要的重排重绘
- 执行时机更精准
七、终极总结
$nextTick
:Vue专用,用于"等DOM更新完再执行"setTimeout
:JS通用,用于"过段时间再执行"
简单记法:
- 只要和Vue的DOM更新有关的延迟,优先用
$nextTick
- 其他纯时间延迟的需求,用
setTimeout
⭐ 写在最后
请大家不吝赐教,在下方评论或者私信我,十分感谢🙏🙏🙏.
✅ 认为我某个部分的设计过于繁琐,有更加简单或者更高逼格的封装方式
✅ 认为我部分代码过于老旧,可以提供新的API或最新语法
✅ 对于文章中部分内容不理解
✅ 解答我文章中一些疑问
✅ 认为某些交互,功能需要优化,发现BUG
✅ 想要添加新功能,对于整体的设计,外观有更好的建议
✅ 一起探讨技术加qq交流群:906392632
最后感谢各位的耐心观看,既然都到这了,点个 👍赞再走吧!