所遇到的响应式问题

问题背景

是在vue2项目升级vue3项目中遇到的,因为升级项目并没有使用vue的Composition api 而是使用Options api,所有复杂类型变量默认使用reactive进行响应式,问题也是从这出现的

  1. 对象数组中,使用索引值更改数据,数据变化了页面没有变化
  • 类似代码 - options api使用的是this指针方式,但是问题是一样的
javascript 复制代码
  arr1:[
    {
        "name": "test",
        "name2": "test2",
        "name3": "test3",
    },
    {
        "name": "test",
        "name2": "test2",
        "name3": "test3",
    }    ]      })    
    arr["arr1"][0] = {"name": "test11","name2": "test22","name3": "test33",}  

这个时候我们从vue3的源代码入手,分析原因,具体只需要看proxy 的 get方法

源代码地址 packages\reactivity\src\baseHandlers.ts

有个BaseReactiveHandler方法

当我们触发get方法时,如果还是复杂类型,需要在调用reactive将其转化成响应式,所以Vue的依赖收集是"按需"的,具有一种懒惰性质,层级较深的复杂类型数据不是在声明式就被全部转化成响应式,而是在获取时逐层转化的

这个时候我们回看我们的问题,当我们执行arr["arr1"][0] = {"name": "test11","name2": "test22","name3": "test33",}的时候,只触发了arr["arr1"],但是再往下层级的并没有被转化成响应式,所以此时我们可以这样去解决

javascript 复制代码
"test11","name2": "test22","name3": "test33",} arr["arr1"] = newArr[0]
  1. 解构失去响应式
javascript 复制代码
const test = reactive({
   arr: { name: '111'}    
})    
   // 解构会失去响应性!    
   const { arr } = test    
   // 修改 user 不会触发界面更新    
   arr.name = '李四'  //界面不更新
   //解决方案   
   // 1、    
   const { arr } = toRefs(test)    
   // 2、尽量不要解构响应式数据
  
  1. Object新增属性不响应
javascript 复制代码
const test = reactive({
  arr: { name: '111'}    
  })
  // 修改 user 不会触发界面更新    
  test.arr.age= 22  
  //界面不更新
  // 解决方法 使用 Object.assign    
  Object.assign(test.arr, { age: 22 })  //  触发更新
 

总结

所以在vue3 开发中,如果使用options api方式,就需要尽量注意多层嵌套对象,如果使用Composition api,尽量使用 ref 去定义变量,并且针对嵌套层级较深的变量最好使用ShallowRef ShallowReactive 用于优化深层嵌套对象的性能问题

相关推荐
暴走的小呆26 分钟前
Vue 2 中 Object 的变化侦测:从 getter/setter 到 Dep、Watcher、Observer
vue.js
奇奇怪怪的29 分钟前
Embedding 模型 10+ 横向评测
前端
陈广亮32 分钟前
Monorepo 从 0 到 1 实操指南 2026 版:pnpm catalogs + Turborepo 2.x + changesets 全链路
前端
子兮曰33 分钟前
OpenMontage 深度解剖:你的 AI 编程助手,其实是个视频工作室
前端·后端·ai编程
敲代码的鱼34 分钟前
PDF 预览与签名批注写回 支持安卓 iOS 鸿蒙 UTS插件
android·前端·ios
英勇无比的消炎药37 分钟前
TinyVue v-auto-tip: 文本超长自动提示的优雅方案
vue.js
子兮曰40 分钟前
前端工具链的「Rust 化」:一场没有赢家的军备竞赛?
前端·后端·rust
Hyyy2 小时前
Function Calling / Tool Use的原理和实现模式
前端·llm·ai编程
爱勇宝2 小时前
从 Ctrl+CV 到 Enter:程序员正在失去什么
前端·后端·程序员
徐小夕2 小时前
我们开源了一款“框架无关”的思维导图编辑器,3分钟集成到任意系统
前端·javascript·github