前端如何虚拟列表优化?

一、为什么要用虚拟列表?

问题本质

当列表数据很多时(如 1w+):

  • DOM 数量过多 → 重排 / 重绘严重

  • 滚动卡顿

  • React diff 变慢

  • 表格(Antd Table)尤其明显

👉 瓶颈不在 JS,而在 DOM


二、虚拟列表核心原理(一句话版)

只渲染"可视区域 + 上下缓冲"的那一小段数据,其它用空白高度"占位"

关键点只有 4 个:

  1. 滚动容器高度固定

  2. 每一项高度固定 or 可计算

  3. 根据 scrollTop 计算 startIndex / endIndex

  4. 用 translateY 或 padding-top 做偏移


三、基础实现原理(固定高度版)

1️⃣ 核心计算公式

复制代码
const itemHeight = 40
const containerHeight = 400

const visibleCount = Math.ceil(containerHeight / itemHeight)
const startIndex = Math.floor(scrollTop / itemHeight)
const endIndex = startIndex + visibleCount + buffer

2️⃣ DOM 结构示意

复制代码
<div class="container" onScroll>
  <div style="height: totalHeight">
    <div style="transform: translateY(offset)">
      {visibleItems.map(render)}
    </div>
  </div>
</div>

3️⃣ React 简化示例

复制代码
const VirtualList = ({ data }) => {
  const containerRef = useRef(null)
  const [scrollTop, setScrollTop] = useState(0)

  const itemHeight = 40
  const containerHeight = 400
  const buffer = 5

  const startIndex = Math.floor(scrollTop / itemHeight)
  const visibleCount = Math.ceil(containerHeight / itemHeight)
  const endIndex = startIndex + visibleCount + buffer

  const visibleData = data.slice(startIndex, endIndex)

  return (
    <div
      ref={containerRef}
      style={{ height: containerHeight, overflow: 'auto' }}
      onScroll={(e) => setScrollTop(e.currentTarget.scrollTop)}
    >
      <div style={{ height: data.length * itemHeight }}>
        <div style={{ transform: `translateY(${startIndex * itemHeight}px)` }}>
          {visibleData.map(item => (
            <div key={item.id} style={{ height: itemHeight }}>
              {item.name}
            </div>
          ))}
        </div>
      </div>
    </div>
  )
}

DOM 永远只有几十个


四、进阶:不定高虚拟列表(真实项目常见)

难点

  • 表格内容不确定

  • 操作列 / 多行文本 / 自动换行

解决思路

方案一(推荐):高度缓存 + 二分查找
复制代码
const heightMap = new Map<index, height>()
  • 首次渲染测量高度

  • 缓存起来

  • 用累计高度数组做滚动定位

👉 react-window、react-virtual 都是这个思路


方案二:估算高度 + 滚动修正
  1. 先用 estimatedHeight

  2. 渲染后真实测量

  3. 修正 scrollTop


五、直接用成熟库(强烈推荐)

1️⃣ react-window(轻量,首选)

复制代码
npm i react-window

import { FixedSizeList as List } from 'react-window'

<List
  height={600}
  itemCount={10000}
  itemSize={40}
  width="100%"
>
  {({ index, style }) => (
    <div style={style}>{data[index].name}</div>
  )}
</List>

👉 性能极好、API 简单


2️⃣ react-virtual(TanStack)

  • 不定高

  • 可横向虚拟

  • 表格/瀑布流都行

    const rowVirtualizer = useVirtualizer({
    count: data.length,
    getScrollElement: () => parentRef.current,
    estimateSize: () => 40,
    })


3️⃣ Ant Design 表格怎么办?

官方推荐组合
  • Table + react-window

  • VirtualTable(社区实现)

Antd v5 已支持 virtual

复制代码
<Table
  columns={columns}
  dataSource={data}
  scroll={{ y: 600 }}
  virtual
/>

⚠️ 列宽、固定列、合并单元格要测试


六、你在性能平台里如何用(结合你实际项目)

你之前提到的场景非常典型:

  • 性能查询列表

  • 性能问题 / 覆盖度表格

  • CPU 架构对比

推荐策略

场景 建议
普通列表 react-window
Antd 表格 Antd v5 virtual 或 react-virtual
图表下方明细 虚拟列表 + memo
操作列多 行高度固定,避免 auto

七、常见坑(非常重要)

❌ 1. 监听 window.scroll

👉 必须用容器滚动

❌ 2. key 用 index

👉 用业务唯一 id

❌ 3. 频繁 setState

👉 requestAnimationFrame / 节流

❌ 4. 虚拟 + 动画

👉 几乎一定卡


八、性能组合拳(虚拟列表 ≠ 万能)

虚拟列表 必须配合

  • React.memo

  • useCallback

  • 列 render 函数拆分

  • 避免匿名函数

  • 避免同步计算


九、总结一句话

虚拟列表的本质不是"快",而是"DOM 数量永远可控"

相关推荐
Easonmax11 小时前
零基础入门 React Native 鸿蒙跨平台开发:7——双向滚动表格实现
react native·react.js·harmonyos
Easonmax11 小时前
零基础入门 React Native 鸿蒙跨平台开发:6——竖向滚动表格实现
react native·react.js·harmonyos
提笔了无痕11 小时前
Web中Token验证如何实现(go语言)
前端·go·json·restful
戌中横11 小时前
JavaScript——Web APIs DOM
前端·javascript·html
Beginner x_u11 小时前
如何解释JavaScript 中 this 的值?
开发语言·前端·javascript·this 指针
HWL567911 小时前
获取网页首屏加载时间
前端·javascript·vue.js
烟锁池塘柳012 小时前
【已解决】Google Chrome 浏览器报错 STATUS_ACCESS_VIOLATION 的解决方案
前端·chrome
速易达网络12 小时前
基于RuoYi-Vue 框架美妆系统
前端·javascript·vue.js
LYS_061812 小时前
RM赛事C型板九轴IMU解算(4)(卡尔曼滤波)
c语言·开发语言·前端·卡尔曼滤波
Easonmax13 小时前
零基础入门 React Native 鸿蒙跨平台开发:8——固定表头和列的复杂表格
react native·react.js·harmonyos