跳到主要内容位置

Vue 3 Watch API 的使用注意事项

Vue 的 composition api 虽然简化了 vue 组件的代码,但是在实际使用过程中仍然需要注意一些问题。这里我把我在体验 vue 3 composition api 的过程中遇到的比较重要的问题整理了一下,尤其是对于有 React 开发经验的人来学习 Vue,会更有帮助。

Vue 的 composition api 虽然简化了 vue 组件的代码,但是在实际使用过程中仍然需要注意一些问题。这里我把我在体验 vue 3 composition api 的过程中遇到的比较重要的问题整理了一下,尤其是对于有 React 开发经验的人来学习 Vue,会更有帮助。

问题

先看问题:在使用 ref 定义响应式状态时,如果传递了一个对象作为参数,那么在使用 watch() 监听这个对象的变化时,根据这个对象的更新方式,watch() 会有不同的反应。

复现过程

假设父组件中用 ref 定义了一个 user 对象,保存用户信息,然后父组件中有个按钮,在点击的时候会把 user.value 替换成一个全新的 user 对象:

<template>
<HelloWorld :user="user" />
<button @click="getNewUser">update user</button>
</template>
<script setup>
import HelloWorld from "./components/HelloWorld.vue";
import { ref } from "vue";

const user = ref({
name: "san",
age: 10,
});

const getNewUser = () => {
user.value = {
name: "abc",
age: Math.random(), // just to make some changes
};
};
</script>

这里简单的用随机数修改 age 的值,以区分不同的对象,然后把 user 传递给 HelloWorld 子组件。在 HelloWorld 子组件中,如果想使用 watch 监听 user 的变化,这时如果把 props.user 传递给 watch 第一个参数的话,那么这个监听在 user 发生变化时是不会重新执行的:

const props = defineProps({
user: Object,
});

watch(props.user, (newv, oldv) => {
console.log(newv, oldv); // 不会执行
});

这是因为如果直接把 props.user 传递给 watch() ,相当于是监听 user 里边的属性变化,但是 user 这个对象本身每次在点击按钮的时候都会发生变化,这个 watch 监听的还是原始的对象,所以会失效,那么要解决这个问题可以使用函数的形式,当函数里依赖的状态发生变化时,watch 会重新执行:

watch(
() => props.user,
(newv, oldv) => {
console.log(newv, oldv); // 会执行
}
);

还有一种方式是使用 toRefs 把 props 中的所有属性都转换为 ref,这样也可以直接监听 props 中的 user 了,因为这时监听的是 user ref 中的 value 属性,user ref 本身不会变化,变化的只是 value 属性的值,所以也可以触发 watch() 的回调函数:

const { user } = toRefs(props);
watch(user, (newv, oldv) => {
console.log(newv, oldv); // 会执行
});

如果在解构赋值的时候不用 toRefs(),那么即便是在 watch() 中使用函数形式也不行:

const { user } = props;
watch(
() => user,
(newv, oldv) => {
console.log(newv, oldv); // 不会执行
}
);

如果父组件在点击按钮的时候,只修改了一下 user 属性值,例如修改一下年龄:

const getNewUser = () => {
user.value.age = Math.random();
};

那么在子组件中,可以直接监听 user 属性:

watch(props.user, (newv, oldv) => {
console.log(newv, oldv); // 会执行
});

因为 user 只是修改了部分属性,引用没有发生变化,所以当 user 的 age 变化时,watch() 也会重新执行。甚至是直接把 user 从 props 解构出来也没有问题:

const { user } = props;
watch(user, (newv, oldv) => {
console.log(newv, oldv); // 会执行
});

好了,这个就是使用 vue watch api 时遇到的问题,在开发中一定要注意。如果有帮助,请三连,想更好的学前端,请关注峰华前端工程师,感谢观看!

提示

一系列的课程让你成为高级前端工程师。课程覆盖工作中所有常用的知识点和背后的使用逻辑,示例全部都为工作项目简化而来,学完即可直接上手开发!

即使你已经是高级前端工程师,在课程里也可能会发现新的知识点和技巧,让你的工作更加轻松!

《React 完全指南》课程,包含 React、React Router 和 Redux 详细介绍,所有示例改编自真实工作代码。点击查看详情。

《Vue 3.x 全家桶完全指南与实战》课程,包括 Vue 3.x、TypeScript、Vue Router 4.x、Vuex 4.x 所有初级到高级的语法特性详解,让你完全胜任 Vue 前端开发的工作。点击查看详情。

《React即时通信UI实战》课程,利用 Storybook、Styled-components、React-Spring 打造属于自己的组件库。

《JavaScript 基础语法详解》本人所著图书,包含 JavaScript 全面的语法知识和新特性, 可在京东、当当、淘宝等各大电商购买