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

解决 ElSelect 数据量大导致加载速度慢

遇到一个性能相关的问题,使用 Element Plus 的 <ElSelect> 组件在数据量很大时,加载速度变慢。

下面简单分析下原因,并提供了一些解决方法。

1. 问题分析

1、大量 DOM 节点渲染

问题:当数据量非常大时,每一个选项都会生成一个 DOM 节点。在 HTML 中,每一个 <option> 元素都需要单独渲染,导致页面需要处理大量 DOM 元素的加载和渲染,影响页面性能。

影响:浏览器在渲染和操作大量 DOM 时效率会降低,导致组件初始化和操作(如滚动、过滤)变慢。

2、Vue 响应式系统的性能瓶颈

问题:Vue 的响应式系统会追踪每个数据项的状态变化。当 <ElSelect> 中的数据量过大时,Vue 的响应式系统需要为每一个 Option 建立响应式追踪,增加内存和计算的开销,尤其是在更新数据、滚动或筛选时,这种情况会更加明显。

影响:响应式追踪在数据项非常多的情况下可能导致浏览器出现卡顿,甚至出现页面响应不及时的情况。

3、事件监听和计算

问题:当 <ElSelect> 中的数据项很多时,每次选择、过滤或输入,都会触发事件监听器和计算操作。如果数据项非常多,这些操作会变得频繁且耗时,增加组件的负担。

影响:页面响应速度降低,用户在操作组件时会感觉到明显的卡顿。

4、过多的无意义的渲染

问题:在默认实现中,<ElSelect> 会一次性渲染所有数据项,不管用户是否在当前视口中看到这些数据。即便用户只滚动一小部分,整个组件仍然会处理所有数据项,导致加载速度慢。

影响:浏览器资源被过度消耗,渲染效率降低,页面加载时间延长。

2. 解决方案

1、使用虚拟滚动

方法:借助虚拟滚动技术(如 Element Plus 的 ElVirtualizedSelect 组件),只渲染当前视口中可见的部分数据。虚拟滚动技术通过动态加载和卸载数据项来减少页面上的 DOM 节点数量。

这种方法能显著减少 DOM 渲染的节点数量和内存占用,提升渲染速度和用户体验。

🌰

<template><!-- 使用虚拟滚动的选择框 --><el-select-v2v-model="selectedValue":options="options"placeholder="请选择"style="width: 200px"/>
</template><script>
import { ref } from 'vue';export default {setup() {// 创建 10,000 条模拟数据const options = ref(Array.from({ length: 10000 }, (_, index) => ({value: index,label: `选项 ${index + 1}`})));const selectedValue = ref(null);return {options,selectedValue};}
};
</script>

效果:显著减少 DOM 中节点数量,提升了渲染性能。对于大数据场景,只有可见选项会被加载和渲染,大大降低了内存和渲染开销。

文档:https://element-plus.org/zh-CN/component/select-v2.html

对比:普通的 <ElSelect> 组件,在最开始渲染全部的 Option 元素。

<template><!-- 使用普通的选择框 --><el-select v-model="value" placeholder="Select" style="width: 240px"><el-optionv-for="item in options":key="item.value":label="item.label":value="item.value"/></el-select>
</template><script>
import { ref } from 'vue'
export default {setup() {// 创建 100 条模拟数据,防止页面卡住const options = ref(Array.from({ length: 100 }, (_, index) => ({value: index,label: `选项 ${index + 1}`})))const selectedValue = ref(null)return {options,selectedValue}}
}
</script>

而 <el-select-v2> 组件只渲染展示的一部分,显而易见的提升了渲染性能。

2、分页加载或懒加载

方法:将数据进行分页或分批次加载。比如,可以设置一个加载阈值,先加载一部分数据项,用户向下滚动到一定程度再加载下一部分数据项。

避免一次性加载大量数据,减少页面初始化时的加载压力。

🌰

<template><el-selectv-model="selectedValue"placeholder="请选择"filterable@visible-change="handleVisibleChange"><el-option v-for="item in options" :key="item.value" :label="item.label" :value="item.value" /></el-select>
</template>
<script>
import { ref, nextTick } from 'vue'
export default {setup() {const options = ref([])const page = ref(1)const selectedValue = ref(null)// 模拟 API 获取分页数据const loadOptions = async () => {const newOptions = await fetchOptions(page.value)options.value.push(...newOptions)page.value++}// 处理滚动事件const handleScroll = (event) => {const { scrollTop, clientHeight, scrollHeight } = event.targetif (scrollTop + clientHeight >= scrollHeight - 10) {loadOptions()}}// 监听下拉框的可见性变化const handleVisibleChange = async () => {await nextTick()const dropdown = document.querySelector('.el-select-dropdown .el-scrollbar__wrap')if (dropdown) {dropdown.addEventListener('scroll', handleScroll)}}// 初始加载loadOptions()return {options,selectedValue,handleVisibleChange}}
}
// 模拟 API 调用,获取分页数据
async function fetchOptions(page) {return Array.from({ length: 10 }, (_, index) => ({value: (page - 1) * 10 + index,label: `选项 ${(page - 1) * 10 + index + 1}`}))
}
</script>

效果:初次加载仅渲染一部分数据,滚动到列表底部时加载更多。通过分页,可以避免一次性加载全部数据,减少页面初始化的负担。

展示:

以此类推,直到数据加载完成后结束。

3、减少不必要的响应式追踪

方法:将不需要响应式的数据项转换为非响应式对象或深度克隆数据。Vue 3 提供了 shallowRef 和 shallowReactive,可用来减少不必要的响应式开销。

效果:降低 Vue 响应式系统的性能开销,提升加载和操作的流畅度。

🌰

<template><el-select v-model="selectedValue" placeholder="请选择"><el-optionv-for="item in nonReactiveOptions":key="item.value":label="item.label":value="item.value"/></el-select>
</template><script>
import { shallowRef, ref } from 'vue';
export default {setup() {// 使用 shallowRef 包装不需要响应式的数据const nonReactiveOptions = shallowRef(Array.from({ length: 1000 }, (_, index) => ({value: index,label: `选项 ${index + 1}`})));const selectedValue = ref(null);return {nonReactiveOptions,selectedValue};}
};
</script>

使用 shallowRef 后,Vue 不会深度监听 nonReactiveOptions 的变化,仅在整个对象被替换时触发重新渲染,这样减少 Vue 对数据的追踪和性能开销。

展示:一次性加载完,但不会跟踪内部变化。

4、减少过度的事件监听

方法:对用户输入和操作添加防抖或节流处理,避免频繁地触发数据项的更新和过滤。比如使用 lodash.debounce 限制输入框触发的过滤频率。

🌰

<template><el-select v-model="selectedValue" filterable @input="onInput" placeholder="请选择"><el-optionv-for="item in filteredOptions":key="item.value":label="item.label":value="item.value"/></el-select>
</template><script>
import { ref, computed } from 'vue';
import debounce from 'lodash/debounce';export default {setup() {const options = ref(Array.from({ length: 1000 }, (_, index) => ({value: index,label: `选项 ${index + 1}`})));const searchQuery = ref('');const selectedValue = ref(null);// 使用防抖处理输入事件const onInput = debounce((value) => {searchQuery.value = value;}, 300);const filteredOptions = computed(() =>options.value.filter((item) =>item.label.includes(searchQuery.value)));return {filteredOptions,selectedValue,onInput};}
};
</script>

效果:只有在输入停止 300 毫秒后,才会触发过滤逻辑,从而避免了输入框内容频繁更新导致的高计算开销。这个方法适用于需要实时过滤的场景。

展示:

总结:

当 Element Plus 的 <ElSelect> 组件加载大量数据时,主要是 DOM 渲染、Vue 响应式追踪和事件计算等导致性能下降。通过使用虚拟滚动、分页加载、减少响应式追踪以及事件防抖等方法,可以显著优化加载性能,使组件在大数据量下也能流畅运行。

相关文章:

解决 ElSelect 数据量大导致加载速度慢

遇到一个性能相关的问题&#xff0c;使用 Element Plus 的 <ElSelect> 组件在数据量很大时&#xff0c;加载速度变慢。 下面简单分析下原因&#xff0c;并提供了一些解决方法。 1. 问题分析 1、大量 DOM 节点渲染 问题&#xff1a;当数据量非常大时&#xff0c;每一个…...

在 CentOS 系统中,您可以使用多种工具来查看网络速度和流量

在 CentOS 系统中&#xff0c;您可以使用多种工具来查看网络速度和流量 在 CentOS 系统中&#xff0c;您可以使用多种工具来查看网络速度和流量1. 使用 iftop安装 iftop使用 iftop 2. 使用 nload安装 nload使用 nload 3. 使用 vnstat安装 vnstat初始化 vnstat查看流量 4. 使用 …...

分布式----Ceph部署

目录 一、存储基础 1.1 单机存储设备 1.2 单机存储的问题 1.3 商业存储解决方案 1.4 分布式存储&#xff08;软件定义的存储 SDS&#xff09; 1.5 分布式存储的类型 二、Ceph 简介 三、Ceph 优势 四、Ceph 架构 五、Ceph 核心组件 #Pool中数据保存方式支持两种类型&…...

使用 PyTorch 实现 AlexNet 进行 MNIST 图像分类

AlexNet 是一种经典的深度学习模型&#xff0c;它在 2012 年的 ImageNet 图像分类比赛中大放异彩&#xff0c;彻底改变了计算机视觉领域的格局。AlexNet 的核心创新包括使用深度卷积神经网络&#xff08;CNN&#xff09;来处理图像&#xff0c;并采用了多个先进的技术如 ReLU 激…...

Python爬虫项目 | 一、网易云音乐热歌榜歌曲

文章目录 1.文章概要1.1 实现方法1.2 实现代码1.3 最终效果 2.具体讲解2.1 使用的Python库2.2 代码说明2.2.1 创建目录保存文件2.2.2 爬取网易云音乐热歌榜单歌曲 2.3 过程展示 3 总结 1.文章概要 学习Python爬虫知识&#xff0c;实现简单的一个小案例&#xff0c;网易云音乐热…...

【Linux】HTTP协议和HTTPS加密

文章目录 HTTP1、概念2、认识URL3、协议格式、请求方法和状态码4、HTTP请求和响应报头5、Cookie和Session HTTPS1、对称和非对称加密2、对称非对称加密安全分析3、证书 HTTP 1、概念 我们在应用层定制协议时&#xff0c;不建议直接发送结构体对象&#xff0c;因为在不同的环境…...

Linux编辑/etc/fstab文件不当,不使用快照;进入救援模式

目录 红帽镜像9救援模式 现象 解决 第一步&#xff1a;修改启动参数以进入救援模式 第二步&#xff1a;进入救援模式、获取root权限、编辑/etc/fstab文件 第三步&#xff1a;编辑好后在重启 下面是ai给的模板 红帽镜像9救援模式 编辑/etc/fstab不当时 17 /dev/nvme0n3p1…...

ubuntu升级postgres

已经有了postgres12&#xff0c;记录一下升级从postgres12升级到15的过程及遇到的一些问题&#xff0c;我没有备份&#xff0c;单纯升级 1、升级过程 深色版本 sudo systemctl stop postgresql 升级PostgreSQL 停止PostgreSQL服务&#xff1a; 停止当前版本的PostgreSQL服务…...

vue2在el-dialog打开的时候使该el-dialog中的某个输入框获得焦点方法总结

在 Vue 2 中&#xff0c;如果你想通过 ref 调用一个方法&#xff08;如 inputFocus&#xff09;来聚焦一个输入框&#xff0c;确保以下几点&#xff1a; 确保 ref 的设置正确&#xff1a;你需要确保在模板中正确设置了 ref&#xff0c;并且它指向了你想要操作的组件或 DOM 元素…...

SpringBoot(十七)创建多模块Springboot项目

在gitee上查找资料的时候,发现有不少Springboot项目里边都是嵌套了多个Springboot项目的。这个玩意好,在协作开发的时候,将项目分成多个模块,有多个团队协作开发,模块间定义标准化通信接口进行数据交互即可。 这个好这个。我之前创建的博客项目是单模块的SpringBoot项目,…...

Vue.js 高质量翻页功能的完整开发指南

文章目录 Vue.js 翻页组件的完整开发与优化指南前言分析分页需求与设计要点基础分页功能的实现分页逻辑 优化分页&#xff1a;封装为组件化设计组件化代码 提升用户体验与性能动态调整每页显示的条目数优化移动端与桌面端的展示高性能翻页策略&#xff1a;按需加载与懒加载提示…...

android dvr黑屏

问题现象&#xff1a;dvr拍摄的图片是黑的&#xff0c;没有buffer数据的。 查看相关的log文件发现&#xff1a; video surface 未释放导致 祥见一下报错信息&#xff1a; 38298 2024-10-16 01:02:51.855 4056 32068 W MediaCodecRenderer: java.lang.IllegalStateExcepti…...

css文字间距撑满横向距离

效果&#xff1a; 代码&#xff1a; 、 text-align:justify;text-align-last: justify;...

【Unity基础】对比OnCollisionEnter与OnTriggerEnter

在Unity中&#xff0c;OnCollisionEnter 和 OnTriggerEnter 是两种用于处理碰撞的回调函数&#xff0c;但它们的工作方式和使用场景有所不同&#xff1a; 1. OnCollisionEnter 触发条件&#xff1a;当一个带有 Collider 组件并且**未勾选“Is Trigger”**的物体&#xff0c;与…...

算法训练(leetcode)二刷第二十五天 | *134. 加油站、*135. 分发糖果、860. 柠檬水找零、*406. 根据身高重建队列

刷题记录 *134. 加油站*135. 分发糖果860. 柠檬水找零*406. 根据身高重建队列 *134. 加油站 leetcode题目地址 当前站点可以剩余油量gas[i] - cost[i]; 将每站的剩余油量求和计算累计剩余油量&#xff0c;总剩余油量小于0&#xff0c;则无法行驶一周。 若在到达某一站时累计剩…...

Springboot 整合 itext 实现PDF文件合并,识别图片则转成PDF拼接

目录 前言一、引用依赖二、使用步骤1.Controller2.Service接口3.实现类三、请求接口及结果前言 本文实现 Springboot 整合 itext 实现PDF文件合并,图片转PDF拼接。 一、引用依赖 <dependency><groupId>com.itextpdf</groupId><artifactId>itext7-co…...

TypeScript 中的 ! 和 ? 操作符

在 TypeScript 中&#xff0c;! 和 ? 是两个非常重要且常用的操作符&#xff0c;分别用于非空断言和可选链操作。下面简单介绍一下二者。 1. 非空断言操作符 ! 1.1 含义 非空断言操作符 !&#xff08;Non-null assertion operator&#xff09;用来告诉 TypeScript 编译器&a…...

开源三代示波器的高速波形刷新方案开源,支持VNC远程桌面,手机,Pad,电脑均可访问(2024-11-11)

说明&#xff1a; 1、本来这段时间是一年一度Hackaday硬件设计开源盛宴&#xff0c;但hackaday电子大赛在去年终结了。所以我开源个我的吧。 2、三代示波器的高速波形刷新方案&#xff0c;前两年就做好了&#xff0c;这两年忙H7-TOOL的更新比较多&#xff0c;三代示波器的更新…...

谷歌推出设备内置人工智能,实时向手机用户发出诈骗电话警报

Google 宣布推出适用于 Android 的新安全功能&#xff0c;可实时防御诈骗和有害应用。 这些功能由先进的设备内置 AI 提供支持&#xff0c;可在不损害隐私的情况下增强用户安全性。 这些新的安全功能首先在 Pixel 上推出&#xff0c;并将很快在更多 Android 设备上推出。 诈…...

AI换人脸facefusion项目口型同步‌API化改造及部署

一. 简介 ‌FaceFusion‌是一款强大的AI换脸软件&#xff0c;它支持图片、视频以及直播换脸&#xff0c;官方将其称为“下一代脸部交换器和增强器”。FaceFusion的最新版本为2.6.1&#xff0c;这个版本在原有基础上增加了更多的模型和高清算法&#xff0c;显著提升了图片和视频…...

为什么92%的研究者搜不到关键书评?Perplexity图书评论搜索的3大认知盲区与实时校准方案

更多请点击&#xff1a; https://codechina.net 第一章&#xff1a;为什么92%的研究者搜不到关键书评&#xff1f; 学术资源检索的失效&#xff0c;往往并非源于信息缺失&#xff0c;而是检索逻辑与出版生态的错位。当前主流学术数据库&#xff08;如Google Scholar、CNKI、JS…...

从芯片手册到PCB:SPL06与MPU9250的I2C实战布线要点与防护设计

从芯片手册到PCB&#xff1a;SPL06与MPU9250的I2C实战布线要点与防护设计 在无人机飞控板的设计中&#xff0c;气压传感器SPL06和九轴传感器MPU9250的稳定工作直接关系到飞行姿态控制的精确性。本文将深入探讨这两个关键传感器在PCB布局中的I2C总线设计要点&#xff0c;以及如何…...

4大技术支柱:构建Pixelle-Video的模块化AI视频生成系统

4大技术支柱&#xff1a;构建Pixelle-Video的模块化AI视频生成系统 【免费下载链接】Pixelle-Video &#x1f680; AI 全自动短视频引擎 | AI Fully Automated Short Video Engine 项目地址: https://gitcode.com/GitHub_Trending/pi/Pixelle-Video 传统视频制作流程需要…...

抖音批量下载工具终极指南:从零开始实现高效无水印下载

抖音批量下载工具终极指南&#xff1a;从零开始实现高效无水印下载 【免费下载链接】douyin-downloader A practical Douyin downloader for both single-item and profile batch downloads, with progress display, retries, SQLite deduplication, and browser fallback supp…...

系统辨识避坑指南:为什么你的最小二乘估计总是不准?从理论到MATLAB仿真的5个常见误区

系统辨识避坑指南&#xff1a;为什么你的最小二乘估计总是不准&#xff1f;从理论到MATLAB仿真的5个常见误区 在系统辨识的实际应用中&#xff0c;许多学习者和初级研发人员都会遇到一个共同的困惑&#xff1a;明明按照教科书上的步骤进行操作&#xff0c;为什么得到的结果却总…...

终极Windows和Office激活指南:KMS智能激活工具三步永久激活方案

终极Windows和Office激活指南&#xff1a;KMS智能激活工具三步永久激活方案 【免费下载链接】KMS_VL_ALL_AIO Smart Activation Script 项目地址: https://gitcode.com/gh_mirrors/km/KMS_VL_ALL_AIO 还在为Windows系统频繁弹出激活提示而烦恼吗&#xff1f;Office突然变…...

OpCore-Simplify:从技术迷宫到一键直达的黑苹果配置革命

OpCore-Simplify&#xff1a;从技术迷宫到一键直达的黑苹果配置革命 【免费下载链接】OpCore-Simplify A tool designed to simplify the creation of OpenCore EFI 项目地址: https://gitcode.com/GitHub_Trending/op/OpCore-Simplify 曾经有多少个夜晚&#xff0c;你对…...

Figma界面3分钟变中文:设计师必备的完整汉化终极指南

Figma界面3分钟变中文&#xff1a;设计师必备的完整汉化终极指南 【免费下载链接】figmaCN 中文 Figma 插件&#xff0c;设计师人工翻译校验 项目地址: https://gitcode.com/gh_mirrors/fi/figmaCN 还在为Figma的英文界面而烦恼吗&#xff1f;作为一名中文设计师&#x…...

FPGA设计实战:别再乱用复位了!同步、异步与异步复位同步释放的Verilog代码避坑指南

FPGA设计实战&#xff1a;复位电路设计的黄金法则与Verilog避坑指南 在FPGA开发的世界里&#xff0c;复位电路就像交响乐团的指挥——它决定了整个系统能否从混沌走向有序。许多工程师往往低估了复位设计的重要性&#xff0c;直到项目后期遭遇难以追踪的亚稳态问题或时序收敛失…...

《流畅的Python》读书笔记03(补充01): 丰富的序列 - Python序列类型核心解析

《流畅的 Python》第 2 章“丰富的序列”系统性地阐述了 Python 序列类型的体系结构、核心操作及其背后的设计哲学。本章内容可归纳为以下四个核心模块&#xff1a; 一、序列类型的分类体系 Python 序列从两个正交维度进行分类&#xff0c;形成了清晰的类型矩阵。 分类维度类…...