Vue3 + Pinia:子组件修改 Pinia 数据,竟然影响了原始数据?

1、业务场景

在组件通信中,我们经常遇到这样的需求:

  • 表格中的每一行都有一个"编辑"按钮
  • 点击按钮,打开弹窗,展示该行数据用于编辑
  • 修改完成后提交,或取消编辑

初始实现代码

1️、父组件中使用 Pinia 设置当前行数据
javascript 复制代码
// store/station.ts
export const useStationStore = defineStore('station', () => {
  const rowData = ref<RowType>({
    name: '',
    id: '',
    // ...
  })

  const setRowData = (row: RowType) => {
    rowData.value = row //  注意这里
  }

  return { rowData, setRowData }
})
2、父组件中点击编辑按钮
ini 复制代码
const edit = (row: RowType) => {
  stationStore.setRowData(row)
  visible.value = true
}
3、子组件中直接使用 rowData
arduino 复制代码
const { rowData } = storeToRefs(useStationStore())

// 在表单中双向绑定使用
<el-input v-model="rowData.name" />

好像到这里一切都没有问题,当我们点击编辑、修改数据的时候,就会发现 表格中的原始数据被修改了,我们分析一下原因

引用传递 + 响应式副作用

ini 复制代码
rowData.value = row

这里做的是引用赋值,也就是说:

  • rowData.valuerow 指向同一块内存
  • 子组件改 rowData.value.xxx,就是在改原始 row.xxx
  • Vue 的响应式系统会让一切看起来都"流畅同步",但这不一定是你想要的!
正确做法:使用深拷贝隔离数据

这里不做第三方库的推荐,自行查找,也不建议自己手写,寻找一个好的轮子会事半功倍。这边主要介绍的是思路。

一些"取巧"的办法

1、如果你的数据只是修第一层,浅拷贝就可以解决 如:

javascript 复制代码
row = JSON.parse((JSON.stringify(row)))
Object.assign(form.value, stationStore.rowData)

2、如果要深层上面的这两个方法就无法保证数据不会被修改

欢迎大家留言探讨你们的实践经验,如有疏漏或不妥之处,欢迎在评论区指出,互相学习,共同进步!

相关推荐
m0_471199635 分钟前
【小程序】订单数据缓存 以及针对海量库存数据的 懒加载+数据分片 的具体实现方式
前端·vue.js·小程序
编程大师哥6 分钟前
Java web
java·开发语言·前端
A小码哥8 分钟前
Vibe Coding 提示词优化的四个实战策略
前端
Murrays8 分钟前
【React】01 初识 React
前端·javascript·react.js
大喜xi11 分钟前
ReactNative 使用百分比宽度时,aspectRatio 在某些情况下无法正确推断出高度,导致图片高度为 0,从而无法显示
前端
helloCat12 分钟前
你的前端代码应该怎么写
前端·javascript·架构
电商API_1800790524712 分钟前
大麦网API实战指南:关键字搜索与详情数据获取全解析
java·大数据·前端·人工智能·spring·网络爬虫
康一夏14 分钟前
CSS盒模型(Box Model) 原理
前端·css
web前端12314 分钟前
React Hooks 介绍与实践要点
前端·react.js
我是小疯子6615 分钟前
JavaScriptWebAPI核心操作全解析
前端