vue.js

关注公众号 jb51net

关闭
首页 > 网络编程 > JavaScript > javascript类库 > vue.js > Vue3 watch和watchEffect使用

Vue3中watch和watchEffect的区别及使用技巧总结

作者:涔溪

在Vue.js里,watch和watchEffect都用于响应式地追踪数据变化并执行相应操作,不过它们在使用方式、应用场景等方面存在差异,这篇文章主要介绍了Vue3中watch和watchEffect区别及使用技巧的相关资料,需要的朋友可以参考下

详细讲解Vue 3 组合式 API 中watchwatchEffect的核心区别,以及它们的实际使用技巧,以便在开发中根据场景正确选择合适的侦听器。

一、先明确两者的核心定位

在 Vue 3 组合式 API 中,watchwatchEffect都是用于监听响应式数据变化并执行副作用的工具,但设计理念和使用场景有明显区别:

二、核心区别对比(表格 + 示例)

对比维度watchwatchEffect
依赖追踪方式显式指定监听源(需明确告诉要监听什么)隐式追踪(自动收集函数内用到的响应式数据)
执行时机默认懒执行(仅当监听源变化时执行)默认立即执行(组件挂载时先执行一次,再监听变化)
新旧值访问支持(回调函数接收新值、旧值参数)不支持(只能访问当前最新值)
监听源类型可监听单个值、多个值、对象属性、计算属性等只能监听函数内直接使用的响应式数据
停止监听返回停止函数,或依赖组件卸载自动停止同左(返回停止函数,组件卸载自动停止)
清除副作用需通过onInvalidate回调(Vue 3.2 + 推荐)同左(回调函数接收onInvalidate参数)

1. 依赖追踪方式对比(最核心区别)

watch:显式指定监听源

必须明确告诉watch要监听哪个 / 哪些响应式数据,未指定的不会被追踪:

<script setup>
import { ref, watch } from 'vue'

const count = ref(0)
const name = ref('Vue')

// 显式监听 count,name 变化不会触发此 watch
watch(count, (newVal, oldVal) => {
  console.log(`count从${oldVal}变成${newVal}`)
})

// 显式监听多个源(数组形式)
watch([count, name], ([newCount, newName], [oldCount, oldName]) => {
  console.log(`count变化:${oldCount}→${newCount},name变化:${oldName}→${newName}`)
})
</script>

watchEffect:隐式追踪依赖

无需指定监听源,函数内用到的所有响应式数据都会被自动追踪:

<script setup>
import { ref, watchEffect } from 'vue'

const count = ref(0)
const name = ref('Vue')

// 自动追踪 count 和 name(函数内用到了这两个响应式数据)
watchEffect(() => {
  console.log(`当前count:${count.value},当前name:${name.value}`)
})
// 首次执行输出:当前count:0,当前name:Vue(立即执行特性)
// 当count或name变化时,会重新执行
</script>

2. 执行时机对比

watch:默认懒执行

只有监听源发生变化时才会执行,组件挂载时不会主动执行:

<script setup>
import { ref, watch } from 'vue'

const count = ref(0)

// 组件挂载时不会执行,只有 count 变化时才执行
watch(count, (newVal) => {
  console.log(`count变化了:${newVal}`)
})

// 若需要立即执行,需手动设置 immediate: true
watch(count, (newVal) => {
  console.log(`count(立即执行):${newVal}`)
}, { immediate: true }) // 组件挂载时执行一次,之后 count 变化再执行
</script>

watchEffect:默认立即执行

组件挂载时会先执行一次(用于收集依赖),之后依赖变化时再执行:

<script setup>
import { ref, watchEffect } from 'vue'

const count = ref(0)

// 组件挂载时立即执行一次,之后 count 变化时再执行
watchEffect(() => {
  console.log(`count:${count.value}`)
})
// 输出:count:0(挂载时)→ count:1(当count++时)
</script>

3. 新旧值访问对比

watch:支持访问新旧值

回调函数的第一个参数是新值,第二个参数是旧值(引用类型需注意浅对比):

<script setup>
import { ref, watch } from 'vue'

const count = ref(0)

watch(count, (newVal, oldVal) => {
  console.log(`旧值:${oldVal},新值:${newVal}`) // 例如:旧值:0,新值:1
})

// 监听对象(需注意:默认是浅监听,对象内部属性变化需开启 deep: true)
const user = ref({ name: 'Vue', age: 3 })
watch(user, (newUser, oldUser) => {
  console.log('user变化了', newUser, oldUser)
}, { deep: true }) // 深度监听对象内部属性
</script>

watchEffect:不支持访问旧值

只能访问当前最新的响应式数据,无法获取变化前的值:

<script setup>
import { ref, watchEffect } from 'vue'

const count = ref(0)

watchEffect(() => {
  // 只能拿到当前值,拿不到变化前的旧值
  console.log(`当前count:${count.value}`)
})
</script>

三、使用技巧与场景选择

1. 什么时候用watch?

实用技巧:

2. 什么时候用watchEffect?

实用技巧:

3. 共同技巧:停止监听

watchwatchEffect都返回一个停止函数,调用后会停止监听(组件卸载时会自动停止,无需手动调用):

<script setup>
import { ref, watch, watchEffect } from 'vue'

const count = ref(0)

// watch 停止监听
const stopWatch = watch(count, (newVal) => {
  console.log(`count:${newVal}`)
})

// watchEffect 停止监听
const stopWatchEffect = watchEffect(() => {
  console.log(`count:${count.value}`)
})

// 手动停止监听(如某个条件满足时)
const stopAll = () => {
  stopWatch()
  stopWatchEffect()
}
</script>

总结

核心区别

  1. 依赖追踪watch显式指定源,watchEffect隐式追踪;
  2. 执行时机watch默认懒执行,watchEffect默认立即执行;
  3. 新旧值watch支持访问新旧值,watchEffect不支持。

场景选择

关键技巧

  1. watch监听对象单个属性时,用函数返回(避免深度监听);
  2. watchEffect需清除副作用时,使用 onInvalidate
  3. 两者都可通过返回的停止函数手动停止监听,组件卸载时自动停止。

到此这篇关于Vue3中watch和watchEffect的区别及使用技巧总结的文章就介绍到这了,更多相关Vue3 watch和watchEffect使用内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

您可能感兴趣的文章:
阅读全文