React项目实战 | 修复Table可展开行,点击一个全部展开

问题分析

Table 组件需要唯一的 rowKey 来正确管理展开状态

原因详解

1. 虚拟DOM diff 算法依赖

React 使用虚拟DOM diff算法来高效更新UI。当Table数据更新时,React需要:

  • 识别哪些行是新增的、哪些是删除的、哪些是更新的
  • 正确保持展开/收起状态
  • 如果没有唯一key,React无法准确追踪每行的状态

2. 展开状态管理

Table组件的展开状态是基于 rowKey 来管理的:

javascript 复制代码
// 内部类似这样的结构
expandedRowKeys = ['id1', 'id3', 'id5']

如果没有唯一id,点击展开时可能出现:

  • 多个行同时展开(主包遇到的bug)
  • 展开状态错乱
  • 无法正确收起

3. 性能优化

唯一key帮助React:

  • 减少不必要的重渲染
  • 提高diff算法效率
  • 准确更新特定行的展开状态

解决方案对比

方案一:使用 uuid

javascript 复制代码
import { v4 as uuidv4 } from 'uuid';

const dataWithId = recordData.list.map(item => ({
  ...item,
  id: uuidv4() // 全局唯一
}));

优点:绝对唯一

缺点:需要额外依赖包

方案二:时间戳+随机数

javascript 复制代码
const dataWithId = recordData.list.map(item => ({
  ...item,
  id: `${Date.now()}-${Math.random().toString(36).substring(2, 9)}`
}));

优点:无依赖

缺点:极低概率重复(通常可接受)

方案三:使用数据中的唯一字段&字段拼接

javascript 复制代码
// 如果数据本身有唯一字段
<Table rowKey="user_id" />

// 或者组合多个字段、data接口数据
const dataWithId = data.map(item => ({
  ...item,
  id: `${item.user_id}-${item.timestamp}`
}));

最佳实践

javascript 复制代码
// 推荐:优先使用数据中的业务ID
// 
const dataWithId = recordData.list.map((item, index) => ({
  ...item,
  id: item.id || item.user_id || `row-${index}-${Date.now()}`
}));

<Table
  rowKey="id"
  columns={columns}
  expandable={{
    expandedRowRender: record => (
      <p style={{ margin: 0 }}>{record.change_content}</p>
    ),
    rowExpandable: record => record.change_content && record.change_content !== ''
  }}
  dataSource={dataWithId}
  pagination={false}
/>

正常表格数据都应有ID,没有去找后端要(狗头

那话又说回来了,真没招了就选择引入UUID或者时间戳+随机数的方法

总结

根本原因 :Table组件依赖唯一的 rowKey 来正确管理每行的展开状态和进行高效的虚拟DOM diff。

修复核心:确保每行数据都有一个唯一标识,让React能够准确追踪和管理每行的状态。

相关推荐
离&染10 小时前
vue.js2.x + elementui2.15.6实现el-select滚动条加载数据
前端·javascript·vue.js·el-select滚动加载
inferno10 小时前
HTML基础(第一部分)
前端·html
kirinlau10 小时前
pinia状态管理在vue3项目中的用法详解
前端·javascript·vue.js
zhuà!11 小时前
腾讯地图TMap标记反显,新增标记
前端·javascript·vue.js
未知原色11 小时前
web worker使用总结(包含多个worker)
前端·javascript·react.js·架构·node.js
ttod_qzstudio11 小时前
CSS改变图片颜色方法介绍
前端·css
curdcv_po11 小时前
我接入了微信小说小程序官方阅读器
前端·微信小程序
程序员鱼皮12 小时前
什么是 RESTful API?凭什么能流行 20 多年?
前端·后端·程序员
www_stdio12 小时前
让大语言模型拥有“记忆”:多轮对话与 LangChain 实践指南
前端·langchain·llm
inferno12 小时前
JavaScript 基础
开发语言·前端·javascript