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

vue3中watch监听不是你想的那样简单的更多相关文章

  1. Android中如何监听GPS开启和关闭

    转自 chenming 原文 Android中如何监听GPS开启和关闭   摘要: 本文简单总结了如何监听GPS开关的小技巧 有时需要监听GPS的开关(这种需求并不多见).实现的思路是监听代表 GPS ...

  2. 新建Oracle数据库时,提示使用database control配置数据库时,要求在当前oracle主目录中配置监听程序

    新建一个oracle数据库时,当提示使用database control配置数据库时,要求在当前oracle主目录中配置监听程序等字样的时候,问题是那个监听的服务没有启动,解决方法如下: 打开cmd命 ...

  3. Nginx 中 fastcgi_pass 监听端口 unix socket和tcp socket差

    Nginx 中 fastcgi_pass 监听端口 unix socket和tcp socket差别   Nginx连接fastcgi的方式有2种:unix domain socket和TCP,Uni ...

  4. oracle 11g在安装过程中出现监听程序未启动或数据库服务未注册到该监听程序

    15511477451 原文 oracle 11g在安装过程中出现监听程序未启动或数据库服务未注册到该监听程序? 环境:win7 64位系统.oracle11g数据库 问题描述:在win7 64位系统 ...

  5. Android 关于ListView中按钮监听的优化问题(方法二)

    关于ListView中按钮监听的优化问题(方法一)地址: http://www.cnblogs.com/steffen/p/3951901.html 之前的方法一,虽然能够解决position的传递, ...

  6. Android 关于ListView中按钮监听的优化问题(方法一)

    在Android应用开发过程中经常会用到ListView,并且每次在item中都要对点击事件进行监听.在给按钮添加OnClickListener时,一般会下意识的在getView()中找到每一个But ...

  7. spring中配置监听队列的MQ

    一.spring中配置监听队列的MQ相关信息注:${}是读取propertites文件的常量,这里忽略.绿色部分配置在接收和发送端都要配置.  <bean id="axx" ...

  8. Android零基础入门第34节:Android中基于监听的事件处理

    原文:Android零基础入门第34节:Android中基于监听的事件处理 上一期我们学习了Android中的事件处理,也详细学习了Android中基于监听的事件处理,同时学会了匿名内部类形式,那么本 ...

  9. angularjs中$watch监听model(对象属性、对象)变化

    昨天看了一下教学视频,学到了有关$watch的用法,想到最近做的一个页面中有个select下拉选项(select中的值变化了,则后面input中的值也跟着相应的变化),不知是否可以使用$watch来代 ...

  10. ALERT日志中常见监听相关报错之中的一个:ORA-609错误的排查

    參考MOS文档有: Troubleshooting Guide ORA-609 : Opiodr aborting process unknown ospid (文档 ID 1121357.1) Al ...

随机推荐

  1. 标准物模型:设备无缝对接,IOT界的福音

    摘要:信息模型是解决IoT产业发展一系列挑战的关键,在信息模型的基础上可以推进行业标准/架构的统一,进而实现产业链生态的协同. 本文分享自华为云社区<[云驻共创]标准物模型,物联网的福音> ...

  2. 加快云原生技术转型, 智能调度登陆华为云DevOps: 增速,节源

    摘要:本文将探讨智能资源调度在华为云DevOps上的应用与实践. 本文分享自华为云社区<加快云原生技术转型, 智能调度登陆华为云DevOps: 增速,节源>,作者: DevAI. 1. 背 ...

  3. iOS加固保护新思路

    ​ 技术简介 前言 iOS加固保护是基于虚机源码保护技术,针对iOS平台推出的下一代加固产品.可以对iOS APP中的可执行文件进行深度混淆.加固,并使用独创的虚拟机技术对代 码进行加密保护,使用任何 ...

  4. 1个案例读懂——游戏产品如何用 A/B 测试做增长

    更多技术交流.求职机会,欢迎关注字节跳动数据平台微信公众号,回复[1]进入官方交流群 随着国内游戏用户数量趋于饱和,中国游戏产业也从高速成长期逐渐转型,市场成熟度提升,竞争趋于精细化. 随着游戏出海以 ...

  5. 【主流技术】聊一聊 Redis 的基本结构和简单应用(一)

    目录 前言 一.String 类型 二.List 类型 三.Hash 类型 四.Set 结构 五.Sort Set (Zset)结构 六.文章小结 前言 Redis 是目前互联网后端的热门中间件之一, ...

  6. xv6book阅读 chapter1

    xv6book主要研究了xv6如何实现它的类Unix接口,但是其思想和概念不仅仅适用于Unix.任何操作系统都必须将进程多路复用到底层硬件上,相互隔离进程,并提供受控制的进程间通信机制. 1 了解xv ...

  7. 自己实现的一个简单的C# IOC 容器

    IService接口,以实现服务的启动.停止功能: using System; using System.Collections.Generic; using System.Linq; using S ...

  8. 3 分钟创建 Serverless Job 定时获取新闻热搜

    不用掏手机,不用登微博,借助 SAE 定时任务就可以实现每小时获取实时新闻热搜!SAE 场景体验火热开启中,参与还可领好礼! Job 作为一种运完即停的负载类型,在企业级开发中承载着丰富的使用场景.S ...

  9. vue+elementUI+WebSocket接收后台实时消息推送

    vue+elementUI+WebSocket接收后台实时消息推送 https://blog.csdn.net/weixin_40888956/article/details/105971432?ut ...

  10. freeswitch带媒体压力测试方案

    概述 原本的计划是使用sipp完成带媒体压力测试,但是实际测试过程中发现sipp的媒体处理功能有问题(也有可能是我使用的姿势不对). sipp在带媒体的情况下(600路并发开始),出现大量的不响应和响 ...