首页 > 其他分享 >前端长列表优化

前端长列表优化

时间:2024-01-18 11:34:09浏览次数:29  
标签:vue scroller 渲染 前端 高度 virtual 列表 优化

一、场景引入

长列表网页相信大多数开发者都遇到过,在DOM元素过多的情况下,浏览器渲染会很慢,非常影响用户体验。
因此我们会经常采用虚拟滚动、分页、上拉加载更多等不同的方式来进行优化,这些方式的思想都是一样的,都是只渲染可见区域,等用户需要时再加载更多的内容。

二、解决方案

1.css属性

content-visibility是一个css属性,它控制一个元素是否呈现其内容,能让用户潜在地控制元素的呈现。用户可以使用它跳过元素的呈现(包括布局和绘制),直到用户需要为止,让页面的初始渲染得到极大的提升。

content-visibility属性有三个可选值:

  • visible: 默认值。对布局和呈现不会产生什么影响。
  • hidden: 元素跳过其内容的呈现。用户代理功能(例如,在页面中查找,按Tab键顺序导航等)不可访问已跳过的内容,也不能选择或聚焦。类似于对其内容设置了display: none属性。
  • auto: 对于用户可见区域的元素,浏览器会正常渲染其内容;对于不可见区域的元素,浏览器会暂时跳过其内容的呈现,等到其处于用户可见区域时,浏览器在渲染其内容。

浏览器兼容性(chrome85):

存在问题:

当元素的部分内容如<img />标签这种,元素的高度是有图片内容决定的,因此在这种情况下,如果使用content-visibility,则可见视图外的img初始未渲染,高度为0,随着滚动条向下滑动,页面高度增加,会导致滚动条的滚动有问题。

解决办法:

如果在已知元素高度的情况下,可以使用contains-intrinsic-size属性,这会给内容附一个初始高度值。(如果高度不固定也可以附一个大致的初始高度值,会使滚动条问题相对减少)。 总结:

content-visibility是一个非常实用的CSS属性,通过一行CSS可以代替虚拟滚动、上拉加载更多等多种长列表渲染优化方式。

虽然其兼容性现在不是很好,但是相信不久的将来这并不是问题。现在来看是部分场景下它对浏览器的滚动条影响问题,如果你的列表项高度相同,那么可以通过contain-intrinsic-size来设置一个初始高度解决。如果列表项高度不固定而又非常重视用户的滚动条体验,那么不建议使用此属性。

2.虚拟列表

虚拟列表是指对列表的 可视区域 进行渲染,对 非可见区域 不渲染或部分渲染,从而极大提高渲染性能的一种技术。

原理:

 

  • 可视区容器:可以看作是在最底层,容纳所有元素的一个盒子。

  • 可滚动区域:可以看作是中间层,假设有 10000 条数据,每个列表项的高度是 50,那么可滚动的区域的高度就是 10000 * 50。这一层的元素是不可见的,目的是产生和真实列表一模一样的滚动条。

  • 可视区列表:可以看作是在最上层,展示当前处理后的数据,高度和可视区容器相同。可视区列表的位置是动态变化的,为了使其始终出现在可视区域。

理解以上概念之后,我们再看看当滚动条滚动时,我们需要做什么:

  1. 根据滚动距离和 item 高度,计算出当前需要展示的列表的 startIndex
  2. 根据 startIndex 和 可视区高度,计算出当前需要展示的列表的 endIndex
  3. 根据 startIndexendIndex 截取相应的列表数据,赋值给可视区列表,并渲染在页面上
  4. 根据滚动距离和 item 高度,计算出可视区列表的偏移距离 startOffset,并设置在列表上

 搞清原理,手写一个虚拟列表亦可。

目前虚拟列表已经有很多知名的库,如 vue-virtual-scroller、vue-virtual-scroll-list、react-virtualized 等

3.虚拟列表之 vue-virtual-scroller

安装

npm install --save vue-virtual-scroller

main.js引入

import "vue-virtual-scroller/dist/vue-virtual-scroller.css";
import VueVirtualScroller from "vue-virtual-scroller";
 
Vue.use(VueVirtualScroller);
RecycleScroller组件:
适用于列表每一项高度确定的情况,高度可设置成相同,也可单独配置每一项高度。
<RecycleScroller
    class="scroller"
    :items="list"
    :item-size="32"
    key-field="id"
    v-slot="{ item }"
  >
    <div class="user">
      {{ item.name }}
    </div>
  </RecycleScroller>
DynamicScroller组件:
适用于列表每一项高度不确定的情况。
<DynamicScroller
    :items="items"
    :min-item-size="54"
    class="scroller"
  key-field="id"
  >
    <template v-slot="{ item, index, active }">
      <DynamicScrollerItem
        :item="item"
        :active="active"
        :size-dependencies="[
          item.message,
        ]"
        :data-index="index"
      >
        <div class="avatar">
          <img
            :src="item.avatar"
            :key="item.avatar"
            alt="avatar"
            class="image"
          >
        </div>
        <div class="text">{{ item.message }}</div>
      </DynamicScrollerItem>
    </template>
  </DynamicScroller>

使用中常见问题:

问题一:

Error: Key is undefined on item (keyField is 'id')

keyField的参数设置不对,导致找不到唯一id,需要设置填充数据的唯一key

问题二:

Error: Rendered items limit reached

外层div,class="scroller"应该设置固定高度,不然虚拟列表不起作用

问题三:虚拟列表只显示了一部分

这里可能有两个原因引起:
1、内容的高度低于min-item-size参数

2、没有引入样式

import 'vue-virtual-scroller/dist/vue-virtual-scroller.css'

问题四:有空白的Item

有空白item的原因就是列表数据里面数据有重复的key,id不唯一

 

参考:https://juejin.cn/post/7036152951399776264?searchId=20240117143213A43B8428EFCFBD1E1B0F

 

标签:vue,scroller,渲染,前端,高度,virtual,列表,优化
From: https://www.cnblogs.com/younghxp/p/17972155

相关文章

  • 利用JVS低代码引擎实现列表页和表单按钮的显隐控制
    在开发中,控制页面的元素(如按钮)的可见性是很常见的需求。特别是在需要根据特定字段的值来隐藏或显示某些元素时,例如在列表页和表单中。这样的需求在许多应用程序中都有出现,无论是电子商务平台、管理系统还是其他类型的软件。本文将深入探讨JVS低代码列表引擎中如何根据字段的值来控......
  • 行云部署前端架构解析-前言 | 京东云技术团队
    一个简单的自我介绍项目规模截止目前上万次代码提交,总代码行数1 超过21万行,其中人工维护的代码超过13万行,近千个文件。前端线上服务直接对接的后端服务,达十多个。跟很多应用一样,它有行云的入口,也有独立的服务,还有单独的插件接口它是行云的子应用,也是其它应用的主应......
  • Stream (是异步版本的列表)、StreamBuilder(局部数据更新)
    Stream流Stream的字面意思是水流,Stream不像Future那样只会在未来获取一个值,它可以异步获取0个或者多个值。如果说Future是一个异步版本的int或者String,Stream则更像是异步版本的列表,List,List,列表里面可能会有0个或者多个元素。classMyHomePageextendsStatefulWidget{......
  • LIS问题的优化
    普通的LIS问题的时间复杂度是\(O(n^2)\),瓶颈主要是在方程\(f[i]=1+max(f[j])\),其中\(1≤j<i\)且\(a[j]<a[i]\)中寻找\(j\)上我们尝试用贪心优化,这里的\(j\)就是小于\(i\)的比\(a[i]\)小的且\(f[j]\)最大的\(j\)根据贪心原则,假设当前循环到了\(i\)(还没有开始处理),我们用\(h[k]\)表......
  • ArkTS待办列表清单【代码可执行】
    共分为三个代码文件:/*定义APP中需要用到的模型Models.ets*/exportclassTaskDataModel{privatetasks:Array<string>=["早起早读","准备早餐","阅读周易","学习实践ArkTS","玩游戏放松一下","准备午饭",......
  • sql语句优化
    SELECTt.ID,t.ProjectNumber,t.ProjectRequirement,t.ProjectDescrip......
  • [前端] 重排和重绘
    网页的生成过程解析HTML,生成DOM树。解析CSS,生成CSSOM树。结合DOM树和CSSOM树,为每一个节点计算CSS属性,生成渲染树,RenderTree。生成布局(Flow),计算渲染树上所有节点的位置。将布局绘制(Paint)到屏幕上。布局生成和绘制的过程就是渲染。网页生成的时候至少渲染一次。用户交互......
  • 千万级数据深分页查询SQL性能优化实践
    一、系统介绍和问题描述如何在Mysql中实现上亿数据的遍历查询?先来介绍一下系统主角:关注系统,主要是维护京东用户和业务对象之前的关注关系;并对外提供各种关系查询,比如查询用户的关注商品或店铺列表,查询用户是否关注了某个商品或店铺等。但是最近接到了一个新需求,要求提供查询关注对......
  • 千万级数据深分页查询SQL性能优化实践
    一、系统介绍和问题描述如何在Mysql中实现上亿数据的遍历查询?先来介绍一下系统主角:关注系统,主要是维护京东用户和业务对象之前的关注关系;并对外提供各种关系查询,比如查询用户的关注商品或店铺列表,查询用户是否关注了某个商品或店铺等。但是最近接到了一个新需求,要求提供查询关注......
  • (Python)每日代码||2024.1.17||函数中给列表形参默认值时,该默认列表在函数中的改变会
    deff(x,li=[1]):print(id(li))li.append(x)print(li)f('a')#第一次调用函数print()f('b')#第二次调用函数print()f('a',[])#第三次调用函数print()f('b',[2,2])#第四次调用函数print()f('a')#第五次调用函数'''输出14......