渲染十万条数据的方法之分批渲染

2024-08-28 23:36

本文主要是介绍渲染十万条数据的方法之分批渲染,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言:前端渲染海量数据列表的情况可能不多,因为很多列表都采用了分页,但是在读写Excel表数据这种情况下,处理海量数据还是有可能会用到。渲染十万条数据,可以体现前端开发同学处理高性能渲染的能力,也是面试常常问到的问题。

对于一次性插入大量数据的情况,一般有两种做法:

  • 分批渲染
  • 虚拟列表

本次对使用 分批渲染 的方式进行实现

分批渲染

React代码实现

  • requestAnimationFrame 是一个浏览器提供的 API,用于在下一次重绘之前执行回调函数。它通常用于优化动画和其他需要高频率更新的操作
  • generateUniqueKey 可以生成唯一key,使 diff 算法高效,原理参考: js生成唯一标识符(例如key或者id)
  • 由于结合动画帧进行递归分批渲染,数据量大,会处理比较长的时间,所以不要忘记处理内存泄漏,在页面卸载的时候用一个开关去跳出可能尚存的递归
import { useCallback, useEffect, useRef, useState } from "react";interface DataItem {key?: string;slogan: string;bgColor: string;
}/*** 生成唯一 key,这里使用时间戳 + 随机数* 你也可以引入第三方库,如 uuid 或 nanoid,但这里为了减少依赖,直接使用 JS 生成* @returns*/
const generateUniqueKey = () => {return `${new Date().getTime()}-${Math.random().toString(36).substr(2, 9)}`;
};const DATA_LIST = [{ slogan: "我爱学友", bgColor: "green" },{ slogan: "我爱德华", bgColor: "blue" },{ slogan: "我爱黎明", bgColor: "red" },
];const BatchPage = () => {const [list, setList] = useState<DataItem[]>([]);// 用于组件卸载后,清除异步操作,防止内存泄漏const isUnmountedRef = useRef<boolean>(false);// 默认 batchSize = 100,即时间分片的每片为 100,每个动画帧渲染 100 条数据,可以根据实际情况调整const renderBatch = useCallback((restList: DataItem[], existingList: DataItem[], batchSize = 100) => {if (!Array.isArray(restList) ||!restList?.length ||isUnmountedRef.current)return;// splice 除了改变原始数组,还会返回删掉的数组const addList = restList.splice(0, batchSize);// requestAnimationFrame 是一个浏览器提供的 API,用于在下一次重绘之前执行回调函数。它通常用于优化动画和其他需要高频率更新的操作。requestAnimationFrame(() => {const newList = [...existingList, ...addList];setList(newList);renderBatch(restList, newList, batchSize);});},[]);const handleClick = useCallback(() => {const jsMakeDataStartTime = new Date().getTime();console.log("点击按钮时间戳-------->", jsMakeDataStartTime);const data: DataItem[] = [];// 模拟生成 10 万条数据for (let index = 0; index < 100000; index++) {const item = DATA_LIST[Math.floor(Math.random() * DATA_LIST.length)];data.push({key: generateUniqueKey(),...item,});}const jsMakeDataEndTime = new Date().getTime();console.log("JS生成数据时间戳------>", jsMakeDataEndTime);console.log("JS生成数据时间间隔---->",jsMakeDataEndTime - jsMakeDataStartTime);renderBatch(data, []);}, [renderBatch]);useEffect(() => {if (!!list?.length)console.log("数据变化时间戳和长度-->",new Date().getTime(),list?.length);}, [list]);useEffect(() => {isUnmountedRef.current = false;return () => {isUnmountedRef.current = true;};}, []);// 本应使用【className + 引入样式文件】的方式,但为了直观,这里直接使用style演示return (<divstyle={{width: "100vw",height: "100vh",display: "flex",flexDirection: "column",flexWrap: "nowrap",}}><buttonstyle={{ padding: "12px", color: "white", backgroundColor: "black" }}onClick={handleClick}>生成海量数据&渲染</button><div style={{ flex: "1", overflowY: "auto" }}>{Array.isArray(list) &&list.map((item) => {return (<divkey={item?.key}style={{backgroundColor: item?.bgColor,marginTop: "6px",color: "white",}}>{item?.slogan}</div>);})}</div></div>);
};export default BatchPage;

实际效果

在这里插入图片描述

这篇关于渲染十万条数据的方法之分批渲染的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1116203

相关文章

MyBatis-Plus通用中等、大量数据分批查询和处理方法

《MyBatis-Plus通用中等、大量数据分批查询和处理方法》文章介绍MyBatis-Plus分页查询处理,通过函数式接口与Lambda表达式实现通用逻辑,方法抽象但功能强大,建议扩展分批处理及流式... 目录函数式接口获取分页数据接口数据处理接口通用逻辑工具类使用方法简单查询自定义查询方法总结函数式接口

MySQL深分页进行性能优化的常见方法

《MySQL深分页进行性能优化的常见方法》在Web应用中,分页查询是数据库操作中的常见需求,然而,在面对大型数据集时,深分页(deeppagination)却成为了性能优化的一个挑战,在本文中,我们将... 目录引言:深分页,真的只是“翻页慢”那么简单吗?一、背景介绍二、深分页的性能问题三、业务场景分析四、

JAVA中安装多个JDK的方法

《JAVA中安装多个JDK的方法》文章介绍了在Windows系统上安装多个JDK版本的方法,包括下载、安装路径修改、环境变量配置(JAVA_HOME和Path),并说明如何通过调整JAVA_HOME在... 首先去oracle官网下载好两个版本不同的jdk(需要登录Oracle账号,没有可以免费注册)下载完

Java中读取YAML文件配置信息常见问题及解决方法

《Java中读取YAML文件配置信息常见问题及解决方法》:本文主要介绍Java中读取YAML文件配置信息常见问题及解决方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要... 目录1 使用Spring Boot的@ConfigurationProperties2. 使用@Valu

Java 方法重载Overload常见误区及注意事项

《Java方法重载Overload常见误区及注意事项》Java方法重载允许同一类中同名方法通过参数类型、数量、顺序差异实现功能扩展,提升代码灵活性,核心条件为参数列表不同,不涉及返回类型、访问修饰符... 目录Java 方法重载(Overload)详解一、方法重载的核心条件二、构成方法重载的具体情况三、不构

SQL中如何添加数据(常见方法及示例)

《SQL中如何添加数据(常见方法及示例)》SQL全称为StructuredQueryLanguage,是一种用于管理关系数据库的标准编程语言,下面给大家介绍SQL中如何添加数据,感兴趣的朋友一起看看吧... 目录在mysql中,有多种方法可以添加数据。以下是一些常见的方法及其示例。1. 使用INSERT I

Python中反转字符串的常见方法小结

《Python中反转字符串的常见方法小结》在Python中,字符串对象没有内置的反转方法,然而,在实际开发中,我们经常会遇到需要反转字符串的场景,比如处理回文字符串、文本加密等,因此,掌握如何在Pyt... 目录python中反转字符串的方法技术背景实现步骤1. 使用切片2. 使用 reversed() 函

Python中将嵌套列表扁平化的多种实现方法

《Python中将嵌套列表扁平化的多种实现方法》在Python编程中,我们常常会遇到需要将嵌套列表(即列表中包含列表)转换为一个一维的扁平列表的需求,本文将给大家介绍了多种实现这一目标的方法,需要的朋... 目录python中将嵌套列表扁平化的方法技术背景实现步骤1. 使用嵌套列表推导式2. 使用itert

Python使用vllm处理多模态数据的预处理技巧

《Python使用vllm处理多模态数据的预处理技巧》本文深入探讨了在Python环境下使用vLLM处理多模态数据的预处理技巧,我们将从基础概念出发,详细讲解文本、图像、音频等多模态数据的预处理方法,... 目录1. 背景介绍1.1 目的和范围1.2 预期读者1.3 文档结构概述1.4 术语表1.4.1 核

Python使用pip工具实现包自动更新的多种方法

《Python使用pip工具实现包自动更新的多种方法》本文深入探讨了使用Python的pip工具实现包自动更新的各种方法和技术,我们将从基础概念开始,逐步介绍手动更新方法、自动化脚本编写、结合CI/C... 目录1. 背景介绍1.1 目的和范围1.2 预期读者1.3 文档结构概述1.4 术语表1.4.1 核