欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 文旅 > 艺术 > 解决 ElSelect 数据量大导致加载速度慢

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

2025/2/22 2:04:33 来源:https://blog.csdn.net/weixin_52648900/article/details/143712533  浏览:    关键词:解决 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 响应式追踪和事件计算等导致性能下降。通过使用虚拟滚动、分页加载、减少响应式追踪以及事件防抖等方法,可以显著优化加载性能,使组件在大数据量下也能流畅运行。

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com

热搜词