vue3中watch监听不是你想的那样简单

简介: vue3中watch监听不是你想的那样简单

vue3 中watch监听数组,数组变化后未触发回调

今天发生了一个很神奇的现象,就是我使用watch监听数组时。
被监听的数组已经发生了变化。但是没有触发回调操作。
当时的我感到很疑惑? 不应该呀? vue2都是可以的。
vue3 咋个不行了。

我是这样操作的-watch回调并没有触发

<script setup lang="ts">
import { reactive, watch } from 'vue';
let dataObj=reactive({
  list:[{name:'张三',age:30}]
})
// 监听的是一个数组
watch(()=>dataObj.list,(newValue,oldValue)=>{
  console.log('新值',newValue)
  console.log('旧值',oldValue)
})
//点击按钮的时候,数组的值回发生变化.但是并没有触发watch的回调操作。
//什么鬼?
const changeValueHandler=()=>{
  dataObj.list.push({name:'李四', age:32})
}
</script>
<template>
  <div>
    <p>list的数据</p>
    <div>{{ dataObj.list}}</div>
    <button @click="changeValueHandler">改变值</button>
  </div>
</template>

为什么不会触发回调

经过我的查阅信息,资料。
原来vue3的监听数组发生了变化。
如果我们监听的是一个数组,只有当数组被替换时才会触发回调。
如果我们需要在数组新增时触发回调,必须指定 deep 选项。
我们找到了不会触发回调的原因。那接下来就好处理这个问题了
// 监听的是一个数组,指定 deep
watch(()=>dataObj.list,(newValue,oldValue)=>{
  console.log('新值',newValue)
  console.log('旧值',oldValue)
},{deep:true})

为什么deep可以解决这个问题呢?

要知道deep为什么可以解决这个问题。
首先我们应该知道deep的作用是什么?
deep的作用我理解的就一句话:
对被被监听的数据进行深度监听,对象内任意一个属性改变都会触发回调。
也就是说:在默认情况下,watch对对象是浅监听的。

说明 watch对对象监听是浅监听的

<script setup lang="ts">
import { reactive, watch } from 'vue';
let dataObj=reactive({
    obj1:{
        name1:{
            name2:{
                name3:'张三'
            },
        },
   }
})
// 监听的是一个对象,当数据发生变化后,watch并没有别监听到
watch(()=>dataObj.obj1,(newValue,oldValue)=>{
    console.log('新值',newValue)
    console.log('旧值',oldValue)
})
//点击按钮的时候,数据发生变化后。watch无法监听。因为watch是浅监听
const changeValueHandler=()=>{
    dataObj.obj1.name1.name2.name3='我把张三的名称改为-李四'
}
</script>
<template>
  <div>
    <p>list的数据</p>
    <div>{{ dataObj.obj1.name1.name2.name3}}</div>
    <button @click="changeValueHandler">改变值</button>
  </div>
</template>

当我们点击按钮 changeValueHandler 的时候,
虽然在视图数据发生了改变。
但是watch并没有监听到,
这也说明watch确实是浅监听的。

当使用 watch 选项侦听数组时,只有在数组被替换时才会触发回调

let dataObj=reactive({
  list:[{name:'张三',age:30}]
})
const changeValueHandler=()=>{
    对数组进行替换,可以被触发,不需要deep选项
    // dataObj.list=[{name:'李四',age:20}] 
    这样修改。需要制定deep选项才会被触发
    // dataObj.list[0].name='李四' 
    进行删除操作,需要制定deep选项才会被触发
    // dataObj.list.splice(0,1) 
    新增操作, 需要制定deep选项才会被触发
    // dataObj.list.push({name:'李四', age:32}) 
}
如果你对被监听的数组进行【修改】的时候,要制定deep选项也可以被触发。
当你对被监听的数组进行【删除】的时候。 需要制定deep 选项才会被触发。
如果你对被监听的数组进行【新增】的时候,需要制定deep选项也可以被触发。
只有进行【替换的时候】,才不需要进行deep选项

watch监听数组-出现新值和旧值是一样

<script setup lang="ts">
import { reactive, watch } from 'vue';
let dataObj=reactive({
  list:[{name:'张三',age:30}]
})
// 监听数据 这里我们发现新值和旧值是一样的
watch(()=>dataObj.list,(newValue,oldValue)=>{
  console.log('新值',newValue)
  console.log('旧值',oldValue)
},{deep:true})
const changeValueHandler=()=>{
  dataObj.list[0].age++
}
</script>
<template>
  <div>
    <p>张三的年龄</p>
    <div>{{ dataObj.list[0]}}</div>
    <button @click="changeValueHandler">改变值</button>
   </div>
</template>

为什么会出现监听的两个值是一样的呢?

看见上面的图片,是不是发现有点不可思议。新值和旧值是一样的。
那为什么会这样呢?
因为对于引用类型,赋值存的是地址,地址指向的是堆。
怎么解决这样的问题了?
我们不去直接监听一个引用类型,而是去监听引用类型中一个具体的值
// 监听监听引用类型中一个具体的值 
watch(()=>dataObj.list[0].age,(newValue,oldValue)=>{
  console.log('新值',newValue)
  console.log('旧值',oldValue)
},{deep:true})

watch停止监听

有的时候,我们可能只会监听一次。
监听之后,我们就需要取消对watch的监听。
我们可以这样操作。
将watch赋值给一个变量,在取消监听的时候调用变量。
let cancelWatch=watch(你的代码)
cancelWatch(); //这个时候无论怎么变化。watch都不会再在对数据进行监听了。

如何停止对watch的监听

<script setup lang="ts">
import { reactive, watch } from 'vue';
let dataObj=reactive({
  list:[{name:'张三',age:30}]
})
// 监听数据
let stopWatch=watch(()=>dataObj.list,(newValue,oldValue)=>{
  console.log('新值',newValue)
  console.log('旧值',oldValue)
},{deep:true})
const changeValueHandler=()=>{
  dataObj.list[0].age++
}
// 停止对 dataObj.list的监听,后续数据变化不会被监听到
const cancelHandler=()=>{
  stopWatch()
}
</script>
<template>
  <div>
    <p>张三的年龄</p>
    <div>{{ dataObj.list[0]}}</div>
    <button @click="changeValueHandler">改变值</button>
    <button @click="cancelHandler">取消监听</button>
  </div>
</template>

尾声

当写完文章的时候,已经是22:56了。太难了!
明天还要去公司调用接口。一个接口调用半天不通。
难死本宝宝了。不说了。去看看git的 cherry-pick。我还没有用过,呜呜。
如果你觉得我写的不错的话,点一下赞。
听说点赞的小哥哥都追到女朋友了,
咦!你不信,不信你给我点赞看一下! 保准你追到到喜欢的Ta

遇见问题,这是你成长的机会,如果你能够解决,这就是收获。

相关文章
|
15天前
|
JavaScript 前端开发 安全
Vue 3
Vue 3以组合式API、Proxy响应式系统和全面TypeScript支持,重构前端开发范式。性能优化与生态协同并进,兼顾易用性与工程化,引领Web开发迈向高效、可维护的新纪元。(238字)
395 139
|
10天前
|
缓存 JavaScript 算法
Vue 3性能优化
Vue 3 通过 Proxy 和编译优化提升性能,但仍需遵循最佳实践。合理使用 v-if、key、computed,避免深度监听,利用懒加载与虚拟列表,结合打包优化,方可充分发挥其性能优势。(239字)
103 1
|
2月前
|
开发工具 iOS开发 MacOS
基于Vite7.1+Vue3+Pinia3+ArcoDesign网页版webos后台模板
最新版研发vite7+vue3.5+pinia3+arco-design仿macos/windows风格网页版OS系统Vite-Vue3-WebOS。
268 11
|
25天前
|
JavaScript 安全
vue3使用ts传参教程
Vue 3结合TypeScript实现组件传参,提升类型安全与开发效率。涵盖Props、Emits、v-model双向绑定及useAttrs透传属性,建议明确声明类型,保障代码质量。
169 0
|
3月前
|
缓存 前端开发 大数据
虚拟列表在Vue3中的具体应用场景有哪些?
虚拟列表在 Vue3 中通过仅渲染可视区域内容,显著提升大数据列表性能,适用于 ERP 表格、聊天界面、社交媒体、阅读器、日历及树形结构等场景,结合 `vue-virtual-scroller` 等工具可实现高效滚动与交互体验。
361 1
|
2月前
|
JavaScript
Vue中如何实现兄弟组件之间的通信
在Vue中,兄弟组件可通过父组件中转、事件总线、Vuex/Pinia或provide/inject实现通信。小型项目推荐父组件中转或事件总线,大型项目建议使用Pinia等状态管理工具,确保数据流清晰可控,避免内存泄漏。
232 2
|
12天前
|
缓存 JavaScript
vue中的keep-alive问题(2)
vue中的keep-alive问题(2)
221 137
|
5月前
|
人工智能 JavaScript 算法
Vue 中 key 属性的深入解析:改变 key 导致组件销毁与重建
Vue 中 key 属性的深入解析:改变 key 导致组件销毁与重建
663 0
|
5月前
|
JavaScript UED
用组件懒加载优化Vue应用性能
用组件懒加载优化Vue应用性能
|
6月前
|
JavaScript 数据可视化 前端开发
基于 Vue 与 D3 的可拖拽拓扑图技术方案及应用案例解析
本文介绍了基于Vue和D3实现可拖拽拓扑图的技术方案与应用实例。通过Vue构建用户界面和交互逻辑,结合D3强大的数据可视化能力,实现了力导向布局、节点拖拽、交互事件等功能。文章详细讲解了数据模型设计、拖拽功能实现、组件封装及高级扩展(如节点类型定制、连接样式优化等),并提供了性能优化方案以应对大数据量场景。最终,展示了基础网络拓扑、实时更新拓扑等应用实例,为开发者提供了一套完整的实现思路和实践经验。
671 77