使用indexOf查找对象结合Pinia持久化引发的问题

使用 indexOf 查找对象结合 Pinia 持久化引发的问题

1. 问题

最近写代码的时候我遇到一个问题,我在开启了数据持久化的 Pinia 仓库里面写了一个类似"点击切换选中状态"的逻辑:如果数组里没有就 push 进去,有了就 splice 掉。代码如下:

tsx 复制代码
// Pinia 仓库逻辑
const setAddonItems = (addon: IStep3) => {
  const index = addons.value.indexOf(addon) // 查找当前点击的对象是否存在
  if (index === -1) {
    addons.value.push(addon) // 不存在则添加
  } else {
    addons.value.splice(index, 1) // 存在则移除
  }
}

这代码单看好像没有什么问题,但是我却遇到一个很奇怪的现象,明明逻辑没问题,为什么刷新页面后 Pinia 就"失忆"了?当我每次在这个代码运行的页面进行刷新的时候,刷新前存进仓库的数据就好像没了一样,setAddonItems 函数在调用的时候,却对 addons 里面存储的数据视而不见,好像里面就是空的一样。我盯着那行 indexOf(addon) 感觉非常疑惑:数据明明就在那儿,为什么 JS 却视而不见?

以上图片就是我当时 localStorage 里面 addons 数据的情况,正确的应该是不论页面刷新几次,内容相同的数据都不会重复出现,但是现在却因为页面刷新而出现了重复。

这是为什么呢?

2. 原因

原因其实很简单,就是 indexOf 对于数据里面对象数据是否存在的判断规则,以及 Pinia 持久化把数据存入 localStoragelocalStorage 的反序列化。

indexOf() 判断对象数据存在的规则

我们都知道,在 JS 中,基本类型是按 比较相等的,而对象是按 内存地址(引用) 比较相等的。

js 复制代码
const a = { id: 1 };
const b = { id: 1 };

console.log(a === b); // 输出 false

虽然 ab 的内容一模一样,但它们在内存中占据了两个不同的位置。indexOf 内部使用的是全等运算符(===),因此它判定这两个对象"不相等"。

持久化造成的"孪生兄弟"错觉(localStorage 反序列化 )

当我们开启了 persist: truePinia 持久化),流程如下:

  1. 存储时: 插件将内存中的 JS 对象通过 JSON.stringify() 转化成字符串存入 localStorage
  2. 刷新页面: 插件从 localStorage 读取字符串,通过 JSON.parse() 还原成对象。

关键点就在这里: JSON.parse() 产生的是一个全新的对象实例,这代表这个新的对象实例的引用地址已经和之前的地址不再一样了。

  • 页面中的数据(items.STEP3): 是我们代码里定义的原始对象。
  • 仓库中的数据(addons): 是从 localStorage 反序列化回来的新对象。

即便它们长得一模一样,但在 JS 眼里,它们是住在不同地址的"孪生兄弟"。因此,indexOf(addon) 永远找不到那个本该存在的对象,导致判断失效。

3. 解决办法

id 判断,而不是对象(简单,改动少)

ts 复制代码
const setAddonItems = (addon: IStep3) => {
  const index = addons.value.findIndex(
    item => item.id === addon.id
  )

  if (index === -1) {
    addons.value.push(addon)
  } else {
    addons.value.splice(index, 1)
  }
}

只存 id 数组(更推荐)

在大型项目中,为了性能和数据一致性,通常不建议在状态仓库中存储整个对象列表

推荐做法:

  1. 仓库只存储选中的 ID(例如 string[]number[])。
  2. 显示逻辑通过 ID 到原始数据源中匹配。
ts 复制代码
const selectedIds = ref<string[]>([])

const toggleAddon = (id: string) => {
  const index = selectedIds.value.indexOf(id) // ID 是基本类型,indexOf 此时是可靠的
  if (index === -1) {
    selectedIds.value.push(id)
  } else {
    selectedIds.value.splice(index, 1)
  }
}

优点:

  • 节省空间: localStorage 存储压力更小。
  • 绝对可靠: 字符串比较不受引用地址影响。
  • 单一数据源: 避免了仓库里的对象属性(如价格、标题)与原始数据源不同步的问题。

4. 总结

这个问题的本质是 JavaScript 对象比较机制与数据持久化存储之间的冲突。刷新页面后,对象引用被重置,导致状态管理失效

正确的解决方案是:

  1. 避免使用 indexOf 比较对象
  2. 使用唯一标识符(如ID)进行比较
    制与数据持久化存储之间的冲突。刷新页面后,对象引用被重置,导致状态管理失效

正确的解决方案是:

  1. 避免使用 indexOf 比较对象
  2. 使用唯一标识符(如ID)进行比较
  3. 在状态管理中只存储必要信息(ID)
相关推荐
Kurisu57514 分钟前
雾锁王国修改器下载2026最新
前端·修改器代码
OxyTheCrack16 分钟前
【Golang】简述make与new内置函数以及两者的区别
开发语言·golang
Rain50926 分钟前
mini-cc 的 MCP 协议:给 AI 装个 USB-C 接口
c语言·开发语言·前端·人工智能·架构·node.js·ai编程
华科大胡子1 小时前
AI开发者的网络卡点:Anthropic连接超时
开发语言·php
磊 子1 小时前
STL无序关联容器—unorded_set+unorded_map
开发语言·c++
向量引擎1 小时前
从零起步,如何打造专属向量引擎 API 中转工作流?
java·服务器·前端
AI人工智能+电脑小能手1 小时前
【大白话说Java面试题 第84题】【Mysql篇】第14题:为什么用 InnoDB 存储引擎的表建议用整型的自增主键?
java·开发语言·数据库·mysql·面试
丷丩1 小时前
MapLibre GL JS第27课:添加COG栅格源
javascript·map·mapbox·maplibre gl js
IT_陈寒2 小时前
Vue这个动态响应坑把我整不会了
前端·人工智能·后端
bestlanzi2 小时前
使用nvm管理node环境
前端·vue.js·npm