当前位置: 首页 > ai >正文

每天一个前端小知识 Day 33 - 虚拟列表与长列表性能优化实践(Virtual Scroll)

虚拟列表与长列表性能优化实践(Virtual Scroll)


🎯 一、背景:为何需要虚拟列表?

在实际业务中,我们经常需要渲染大量数据项,比如:

  • 聊天记录滚动展示
  • 电商商品长列表
  • 数据中心表格(DataGrid)

如果直接将上千条 DOM 节点一次性渲染,会引起:

问题说明
卡顿/掉帧浏览器 DOM 渲染耗时过大,UI thread 被堵塞
内存暴涨节点数量越多,内存占用越高,GC 压力大
页面响应慢任何交互都有延迟,影响用户体验

🧠 二、什么是虚拟滚动(Virtual Scroll)?

虚拟列表是一种“按需渲染”的技术:

只渲染当前可视区域内的少量 DOM 节点,其余的用空白占位撑起滚动高度,从而模拟完整列表滚动。

通过“视口裁剪 + 滚动同步”,将 10000 项渲染的压力降到几十项以内。


🧩 三、虚拟滚动原理简述

📦 示例:

  • 假设每一项高度为 50px
  • 列表有 10000 项,总高度为 500,000px
  • 实际可视区域只能显示 20 项

👉 只渲染这 20 项 DOM 元素
👉 设置容器滚动高度为总高度
👉 根据滚动位置动态更新“可视区域数据 + offset”


📈 可视区域公式计算:

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

🛠 四、手写简易虚拟列表(原生 JS 示例)

<div id="container" style="height:300px; overflow:auto; position:relative;"><div id="phantom"></div><div id="content" style="position:absolute; top:0;"></div>
</div>
const data = Array.from({ length: 10000 }, (_, i) => `Item ${i}`);
const itemHeight = 30;
const container = document.getElementById('container');
const phantom = document.getElementById('phantom');
const content = document.getElementById('content');phantom.style.height = `${data.length * itemHeight}px`;container.addEventListener('scroll', () => {const scrollTop = container.scrollTop;const start = Math.floor(scrollTop / itemHeight);const visibleCount = Math.ceil(container.clientHeight / itemHeight);const end = start + visibleCount;const visibleData = data.slice(start, end);content.style.transform = `translateY(${start * itemHeight}px)`;content.innerHTML = visibleData.map(d => `<div style="height:${itemHeight}px">${d}</div>`).join('');
});

✅ 这样页面只渲染有限 DOM,性能稳定,滚动流畅。


⚙️ 五、常见框架中的虚拟列表解决方案

框架工具/库特点
Reactreact-window, react-virtualized支持列表、表格、无限滚动等
Vuevue-virtual-scroller轻量虚拟滚动组件
通用virtual-scroll, 自研实现可组合封装、按需优化

✅ React 示例(react-window)

import { FixedSizeList as List } from 'react-window';<Listheight={300}itemCount={10000}itemSize={35}width={300}
>{({ index, style }) => <div style={style}>Row {index}</div>}
</List>

💡 六、进阶优化点(加分项)

优化点说明
预加载/缓存策略滚动过程中预先加载上下 10 项,减少空白感
动态高度支持对于不等高元素,使用 IntersectionObserver 动态调整
滚动锚点保持改变列表数据时自动保留原始滚动位置
图片懒加载 + 虚拟滚动组合实现高性能图文流

🧪 七、面试高频问题拆解

📌 Q1:你如何优化一个 1 万条数据的滚动列表渲染?

答:

使用虚拟列表技术,仅渲染可视区域内的 DOM 元素,其余通过容器撑高模拟滚动。可结合 react-window 或 vue-virtual-scroller 来高效实现。还可结合懒加载、IntersectionObserver 等增强体验。

📌 Q2:虚拟列表支持不等高的 item 吗?如何处理?

答:

可以,但较复杂。需要:

  • 使用 ResizeObserverIntersectionObserver 动态记录每项高度
  • 使用映射表缓存高度
  • 更新滚动偏移时进行累加偏移量校正

更推荐高度一致或接近的场景。

📌 Q3:如何避免滚动白屏或闪烁?

答:

  • 设置合理的 buffer 区(上下额外渲染几项)
  • 使用 will-change 预优化 transform
  • DOM diff 使用 key 提升更新效率
  • 避免频繁触发 innerHTML 重排

✅ 总结

能力维度说明
虚拟列表基本原理利用偏移+容器撑高+动态渲染实现
实现方式原生计算/封装组件/react-window/vue-scroller
面试价值性能优化落地实践,体现工程能力
实战场景电商列表、聊天滚动、日志流、表格大数据等
http://www.xdnf.cn/news/15525.html

相关文章:

  • Oracle 关于一些连接故障的总结
  • NumPy 详解
  • 职业发展:把工作“玩”成一场“自我升级”的游戏
  • Web前端性能优化原理与方法
  • 【kubernetes】--安全认证机制
  • xss-labs通关
  • 微服务架构升级:从Dubbo到SpringCloud的技术演进
  • PandaWiki与GitBook深度对比:AI时代的知识管理工具,选谁好?
  • 数据库(five day)——物物而不物于物,念念而不念于念。
  • 自适应哈希索引 和 日志缓冲区
  • 将Android Studio创建的一个apk工程放到Android15源码中构建
  • Jmeter+ant+jenkins接口自动化测试框架
  • docker run elasticsearch 报错
  • Spring之核心容器(IoC,DI,基本操作)详解
  • LeetCode|Day15|125. 验证回文串|Python刷题笔记
  • 912. 排序数组
  • 基于docker的redis集群
  • web前端用MVP模式搭建项目
  • Redisson实现限流器详解:从原理到实践
  • Vue加密文章密码 VuePress
  • 数据结构 双向链表(1)
  • 基于Matlab的四旋翼无人机动力学PID控制仿真
  • PyTorch 参数初始化详解:从理论到实践
  • ZYNQ Petalinux系统FLASH固化终极指南:创新多分区与双系统切换实战
  • 如何区分Bug是前端问题还是后端问题?
  • UE5多人MOBA+GAS 24、创建属性UI(一)
  • 插板式系统的“生命线“:EtherCAT分布式供电该如何实现?
  • 第13章 AB实验平台的建设
  • 解锁高效Excel技能:摆脱鼠标,快速编辑单元格
  • 凯伦股份融合复合瓦:新时代可焊接物理防腐金属屋面系统方案