vue3的watch监听方式

本文详细介绍了如何在Vue3中使用reactive和ref进行响应式编程,包括监听单个属性、多个属性、对象及对象属性的变化,并讨论了深度监听的问题和解决方案。通过实例演示了watch方法的应用,适合理解和实践Vue3的响应式系统。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

template部分

<template>
  <h1>方式一:监听一个属性</h1>
  <p>求和:{{ sum }}</p>
  <button @click="sum++">方式一点我加1</button><br />


  ------------------------------------------------------------------------------------<br />
  <h2>方式二:监听多个属性</h2>
  <p>{{ tipsmessage }}</p>
  <button @click="tipsmessage += 1">点我拼接1</button><br />


  ------------------------------------------------------------------------------------<br />
  <h1>方式三:监听对象</h1>
  <p>姓名:{{ obj.name }}</p>
  <p>年龄:{{ obj.age }}</p>
  <button @click="obj.age++">点我加年龄</button><br />


  ------------------------------------------------------------------------------------<br />
  <h1>方式四:监听对象某一个属性变化</h1>
  <p>薪资:{{ obj.test.salary }}</p>
  <p>工作年限:{{ obj.test.year }}</p>
  <button @click="obj.test.salary += 1000">点我涨薪</button>
  <button @click="obj.test.year++">点我添加工作年限</button>
  <br />
</template>

js部分:
使用reactive和ref混合响应式watch监听示例

 setup() {
    let sum = ref(122);
    let tipsmessage = ref("hello");
    const obj = reactive({
      name: "老王",
      age: "50",
      test: {
        salary: 3000,
        year: 1,
      },
    });
       // 方式一:监听单个基本数据类型(ref)
       		此处代码看对应示例
       		
       // 方式二:监多个基本数据类型(ref)
		   此处代码看对应示例
		   
	   // 方式三:监听对象(reactive)
			此处代码看对应示例
			
	   // 方式四:监听对象中某一个属性变化(reactive)
	   		此处代码看对应示例
	   		
	   // 方式五:监视对象中某几个属性(reactive)
	   		此处代码看对应示例
	   		
	   //  方式六:监视对象中的某一个对象 但我操作这某一对象里面的某一个属性obj.test(reactive)
	   		此处代码看对应示例
	
  return { sum, tipsmessage, obj };

方式一:监听单个基本数据类型(ref)

   watch(sum, (newVal, oldValue) => {
     		 console.log(newVal, oldValue);
    });

方式二:监多个基本数据类型(ref)

   watch([sum, tipsmessage], (newVal, oldValue) => {
      // [122, 'hello1']监听出来结果发现新的值和旧的值都是数组
      console.log(newVal, oldValue);
    });

方式三:监听对象(reactive)

     vue3当前存在问题:
     1.reactive和ref响应式使用proxy代理无法正确监听并获取对象旧的值
     2.强制开启深度监听,关闭无效(deep配置无效)
     
      watch(
      obj,
      (newVal, oldValue) => {
        // newVal=> {name: '老王', age: 52} oldValue=>{name: '老王', age: 52}
        console.log(newVal, oldValue);
      },
      { deep: false }
    );

方式四:监听对象中某一个属性变化(reactive)

 watch(
      () => obj.test.salary,
      (newVal, oldValue) => {
        console.log(newVal, oldValue);
      }
    );

方式五:监听对象中某几个属性(reactive)
监听属性salary和year

 // 薪资和年龄示例
watch([() => obj.test.salary, () => obj.test.year], (newVal, oldValue) => {
      console.log(newVal, oldValue);
      //[4000, 1][3000, 1]
      const arr = ["salary", "year"];
      for (let i = 0; i < newVal.length; i++) {
        if (newVal[i] !== oldValue[i]) {
          // 1年工作经验 3000 每年涨薪1000
          switch (arr[i]) {
            case "salary":
              obj.test.year = (obj.test[arr[i]] - 3000) / 1000 + 1;
              break;
            case "year":
              obj.test.salary = (obj.test[arr[i]] - 1) * 1000 + 3000;
              break;
          }
        }
      }
    });

方式六:监视对象中的某一个对象 但我操作这某一对象里面的某一个属性obj.test(reactive)

注意:这种方式需要开启深度监听deep:true才能监视到某一对象属性的变化,此处deep配置有效

 watch(
      () => obj.test,
      (newVal, oldValue) => {
        console.log(newVal, oldValue, "啊哈哈哈");
        //{salary: 3000, year: 2} {salary: 3000, year: 2} '啊哈哈哈'
      },
      { deep: true }
    );
    return { sum, tipsmessage, obj };
  },
};

注意点: 使用ref的响应式watch监听
1.ref定义的基本类型使用watch监听时候不需要.value
示例:

 watch(sum, (newVal, oldValue) => {
      console.log(newVal, oldValue);
    });
    

2.ref定义对象类型使用watch监听对象时监听的对象外层可以使用2个方式
示例:

 const obj = ref({
      name: "老王",
      age: "50",
      test: {
        salary: 3000,
        year: 1,
      },
    });

(1).使用.value监听

   watch(obj.value, (newVal, oldValue) => {
      console.log(newVal, oldValue);
    });

(2).开启深度监听deep:true

   watch(obj, (newVal, oldValue) => {
      console.log(newVal, oldValue);
    },{deep:true});
### Vue 3 中 `watch` 的使用方法及示例代码 在 Vue 3 中,`watch` 函数用于监听响应式数据的变化,并在数据变化时执行相应的操作。与 Vue 2 中的 `watch` 选项不同,Vue 3 的 `watch` 是一个函数,可以在组合式 API 中灵活使用,适用于 `<script setup>` 和普通 `<script>` 语法[^1]。 #### 监听单个响应式引用(`ref`) 当使用 `ref` 创建响应式数据时,可以通过 `watch` 直接监听该引用,并在其值变化时触发回调。 ```javascript import { ref, watch } from &#39;vue&#39; const count = ref(0) watch(count, (newVal, oldVal) => { console.log(`count 从 ${oldVal} 变为 ${newVal}`) }) count.value++ // 触发 watch 回调 ``` #### 监听响应式对象的某个属性 如果需要监听一个响应式对象的某个属性,可以通过返回该属性的函数作为监听源。 ```javascript import { reactive, watch } from &#39;vue&#39; const state = reactive({ count: 0 }) watch(() => state.count, (newVal, oldVal) => { console.log(`state.count 从 ${oldVal} 变为 ${newVal}`) }) state.count++ // 触发 watch 回调 ``` #### 监听多个数据源 `watch` 支持同时监听多个数据源,只需将它们作为数组传入,并在回调中接收对应的多个新值和旧值。 ```javascript import { ref, watch } from &#39;vue&#39; const count = ref(0) const name = ref(&#39;Vue&#39;) watch([count, name], ([newCount, newName], [oldCount, oldName]) => { console.log(`count 从 ${oldCount} 变为 ${newCount}`) console.log(`name 从 ${oldName} 变为 ${newName}`) }) count.value++ name.value = &#39;Vue 3&#39; // 同时触发 count 和 name 的 watch 回调 ``` #### 深度监听对象或数组 对于响应式对象或数组,如果希望监听其内部变化,需要设置 `deep` 选项为 `true`。 ```javascript import { reactive, watch } from &#39;vue&#39; const state = reactive({ count: 0, user: { name: &#39;Alice&#39; } }) watch(() => state.user, (newVal, oldVal) => { console.log(&#39;user 对象发生变化&#39;) }, { deep: true }) state.user.name = &#39;Bob&#39; // 触发 watch 回调 ``` #### 监听 Props 的变化 在子组件中监听 `props` 的变化时,由于 `props` 是只读的且可能是一个响应式代理,直接监听对象属性可能不会触发回调。此时应使用箭头函数包装 `props` 的值。 ```javascript import { watch } from &#39;vue&#39; export default { props: [&#39;modelValue&#39;, &#39;options&#39;], setup(props) { watch([() => props.modelValue, () => props.options], ([newModel, newOptions]) => { console.log(&#39;modelValue 或 options 发生变化&#39;, newModel, newOptions) }) } } ``` #### 初始执行 `watch` 回调 默认情况下,`watch` 回调只会在监听的数据发生变化时执行。如果希望在初始化时也执行一次回调,可以设置 `immediate: true`。 ```javascript import { ref, watch } from &#39;vue&#39; const message = ref(&#39;Hello&#39;) watch(message, (newVal) => { console.log(&#39;message 的初始值为:&#39;, newVal) }, { immediate: true }) ``` #### 停止监听 `watch` 返回一个用于停止监听的函数,调用该函数可以取消监听。 ```javascript import { ref, watch } from &#39;vue&#39; const count = ref(0) const stop = watch(count, () => { console.log(&#39;count 发生变化&#39;) }) count.value++ // 触发 watch 回调 stop() // 停止监听 count.value++ // 不再触发 watch 回调 ``` --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

追逐梦想之路_随笔

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值