0

0

js如何实现虚拟列表渲染 js虚拟列表性能优化的6个要点

下次还敢

下次还敢

发布时间:2025-06-28 15:15:02

|

290人浏览过

|

来源于php中文网

原创

虚拟列表渲染是一种优化长列表性能的技术,其核心在于仅渲染可视区域内的列表项。1. 监听滚动事件;2. 计算可视区域的起始索引和结束索引;3. 渲染对应索引的列表项;4. 设置占位元素保证滚动条正常显示。为提升性能,可采取以下措施:5. 减少dom操作,复用dom元素;6. 使用requestanimationframe优化ui更新。当列表项高度不一致时,需维护一个高度数组,并通过遍历该数组累加高度来确定起始和结束索引。虚拟列表不适用于列表项数量少、高度频繁变化或需要快速滚动的场景。为避免白屏或闪烁,可采用预渲染、css transform、图片懒加载及双缓冲技术等手段。上述方法共同保障了虚拟列表在复杂场景下的流畅表现。

js如何实现虚拟列表渲染 js虚拟列表性能优化的6个要点

虚拟列表渲染,简单来说,就是只渲染可视区域内的列表项,而不是一次性渲染整个列表。这样可以大幅提升长列表的性能,避免页面卡顿。

js如何实现虚拟列表渲染 js虚拟列表性能优化的6个要点

解决方案

js如何实现虚拟列表渲染 js虚拟列表性能优化的6个要点

核心思路是:监听滚动事件,计算可视区域的起始索引和结束索引,然后只渲染这些索引对应的列表项。

js如何实现虚拟列表渲染 js虚拟列表性能优化的6个要点
  1. 计算可视区域高度: 获取列表容器的高度,例如 containerHeight = container.offsetHeight;

  2. 计算列表项高度: 假设每个列表项高度一致,获取一个列表项的高度,例如 itemHeight = listItem.offsetHeight; 如果列表项高度不一致,需要维护一个高度数组,记录每个列表项的高度。

  3. 计算可视区域内的起始索引和结束索引: 根据滚动条位置和列表项高度,计算出可视区域内的起始索引 startIndex 和结束索引 endIndex。 公式如下:

    startIndex = Math.floor(scrollTop / itemHeight);
    endIndex = Math.min(listData.length - 1, startIndex + Math.ceil(containerHeight / itemHeight));
  4. 渲染可视区域内的列表项: 只渲染 startIndexendIndex 之间的列表项。

  5. 处理滚动条位置: 为了让滚动条正常显示,需要设置列表容器的总高度。 总高度 = 列表项数量 * 列表项高度。 可以通过设置一个空的占位元素的高度来实现。

JS虚拟列表性能优化的6个要点

  1. 减少DOM操作: 这是最关键的优化点。 尽量复用DOM元素,避免频繁创建和销毁DOM节点。 可以维护一个DOM池,从DOM池中获取可用的DOM元素,渲染完成后再放回DOM池。

  2. 使用requestAnimationFrame: 将更新UI的操作放在 requestAnimationFrame 中执行,可以避免阻塞主线程,提高页面流畅度。

    Vondy
    Vondy

    下一代AI应用平台,汇集了一流的工具/应用程序

    下载
  3. 节流或防抖滚动事件: 滚动事件触发频率很高,如果每次触发都重新计算和渲染,会造成性能问题。 可以使用节流或防抖来降低滚动事件的触发频率。

  4. 使用Intersection Observer API: Intersection Observer API 可以更精确地监听元素是否进入可视区域,避免不必要的计算和渲染。 相比于监听滚动事件,Intersection Observer API 的性能更好。

  5. 优化数据结构: 如果列表数据量很大,可以考虑使用分块加载或懒加载的方式,减少初始加载的数据量。 同时,选择合适的数据结构,例如使用Map代替Array,可以提高查找效率。

  6. 使用Web Worker: 将复杂的计算任务放在Web Worker中执行,可以避免阻塞主线程,提高页面响应速度。 例如,计算可视区域内的起始索引和结束索引,可以在Web Worker中执行。

如何处理列表项高度不一致的情况?

如果列表项高度不一致,就不能简单地使用 itemHeight 来计算起始索引和结束索引了。 需要维护一个高度数组,记录每个列表项的高度。 计算起始索引和结束索引时,需要遍历高度数组,累加高度,直到累加的高度超过滚动条位置。

具体步骤如下:

  1. 维护高度数组: 在初始化时,或者在列表项渲染完成后,获取每个列表项的高度,并将其存储到高度数组中。
  2. 计算起始索引: 遍历高度数组,累加高度,直到累加的高度超过滚动条位置。 当前索引就是起始索引。
  3. 计算结束索引: 从起始索引开始,继续遍历高度数组,累加高度,直到累加的高度超过滚动条位置加上可视区域高度。 当前索引就是结束索引。

虚拟列表在哪些场景下不适用?

虽然虚拟列表可以大幅提升长列表的性能,但并非所有场景都适用。

  • 列表项数量很少: 如果列表项数量很少,虚拟列表的优势不明显,甚至可能带来额外的性能开销。
  • 列表项高度动态变化且频繁: 如果列表项高度动态变化且非常频繁,维护高度数组的成本会很高,可能会影响性能。
  • 需要支持快速滚动: 如果需要支持快速滚动,虚拟列表可能会出现白屏或闪烁的情况。 需要进行额外的优化,例如预渲染一部分列表项。

如何避免虚拟列表渲染时的白屏或闪烁问题?

白屏或闪烁问题通常是由于渲染速度跟不上滚动速度造成的。 可以采取以下措施来缓解这个问题:

  • 预渲染: 在可视区域之外,预渲染一部分列表项。 这样可以在滚动时快速显示内容,避免白屏或闪烁。
  • 使用CSS transform: 使用CSS transform 来移动列表项,而不是直接修改DOM元素的top或left属性。 CSS transform 的性能更好。
  • 优化图片加载: 如果列表项包含图片,可以使用懒加载或预加载的方式来优化图片加载。 避免一次性加载所有图片,造成页面卡顿。
  • 使用双缓冲技术: 使用两个缓冲区,一个用于显示,一个用于渲染。 在渲染完成后,将两个缓冲区进行交换。 这样可以避免渲染过程中出现白屏或闪烁。

相关文章

数码产品性能查询
数码产品性能查询

该软件包括了市面上所有手机CPU,手机跑分情况,电脑CPU,电脑产品信息等等,方便需要大家查阅数码产品最新情况,了解产品特性,能够进行对比选择最具性价比的商品。

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
treenode的用法
treenode的用法

​在计算机编程领域,TreeNode是一种常见的数据结构,通常用于构建树形结构。在不同的编程语言中,TreeNode可能有不同的实现方式和用法,通常用于表示树的节点信息。更多关于treenode相关问题详情请看本专题下面的文章。php中文网欢迎大家前来学习。

550

2023.12.01

C++ 高效算法与数据结构
C++ 高效算法与数据结构

本专题讲解 C++ 中常用算法与数据结构的实现与优化,涵盖排序算法(快速排序、归并排序)、查找算法、图算法、动态规划、贪心算法等,并结合实际案例分析如何选择最优算法来提高程序效率。通过深入理解数据结构(链表、树、堆、哈希表等),帮助开发者提升 在复杂应用中的算法设计与性能优化能力。

30

2025.12.22

深入理解算法:高效算法与数据结构专题
深入理解算法:高效算法与数据结构专题

本专题专注于算法与数据结构的核心概念,适合想深入理解并提升编程能力的开发者。专题内容包括常见数据结构的实现与应用,如数组、链表、栈、队列、哈希表、树、图等;以及高效的排序算法、搜索算法、动态规划等经典算法。通过详细的讲解与复杂度分析,帮助开发者不仅能熟练运用这些基础知识,还能在实际编程中优化性能,提高代码的执行效率。本专题适合准备面试的开发者,也适合希望提高算法思维的编程爱好者。

45

2026.01.06

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

766

2023.08.10

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

766

2023.08.10

golang map内存释放
golang map内存释放

本专题整合了golang map内存相关教程,阅读专题下面的文章了解更多相关内容。

77

2025.09.05

golang map相关教程
golang map相关教程

本专题整合了golang map相关教程,阅读专题下面的文章了解更多详细内容。

40

2025.11.16

golang map原理
golang map原理

本专题整合了golang map相关内容,阅读专题下面的文章了解更多详细内容。

67

2025.11.17

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

26

2026.03.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PHP自制框架
PHP自制框架

共8课时 | 0.6万人学习

简单聊聊mysql8与网络通信
简单聊聊mysql8与网络通信

共1课时 | 850人学习

微信小程序开发(网易云音乐)
微信小程序开发(网易云音乐)

共94课时 | 14.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号