jq 下拉加载每次只执行一次_记一次 无限列表 滚动优化

c784df551443cc8b7191eeaedff423c0.png

背景

长列表优化, 是页面性能优化中的一个比较常见的问题,也是面试中的常客。

刚好最近在的项目中, 遇到了一个长列表的性能问题,试过多种方案, 最后得以解决。

今天就给大家分享一下。

正文

场景描述

用户需要批量修改 Product中 sku 的 映射关系,可以选择的 Product 的 数量不限

每一条sku 对应如下结构:

cb3036bde68249505e8ccb4a61379f79.png

因为可以选择的sku数量是不限的, 又不能分页, 只能做到一个列表里。

于是, 长列表出现了。

刚开始的方案是做一个虚拟列表

具体就是通过监听sroll事件,每次滚动后计算一般元素位置(top和height)

然后,通过渲染三屏的方式,把一段数据渲染到页面上。

数据量不多的时候, 没什么问题。

当选择几百上千条sku 的时候, 快速滑动, 就开始出现卡顿。

如图所示:

197d64ab64c1671e2bad9a90e3be7070.gif

快速滚动出现空白

作为对比,看一下优化后的效果:

84893220ff9ed755932fcd10191c6777.gif

优化之后

问题定位

在chrome调试工具下,边拖动列表边观察dom的变化。

发现,dom的卸载/挂载/更新的情况都出奇地慢,鼠标已经停下来,能明显感觉到过一会dom才装载完成,所以很可能是dom的渲染性能问题。

定位到渲染性能有问题的dom身上,即每一个 Item(renderFakeTable)。

使用普通文本代替Item,在同样多数量的列表情况下,简单的dom明显会顺畅很多,但是,仍然会出现空白问题。

继续观察renderFakeTable中的每一个元素(可以借用devTools Profiler)。

最简单粗暴的方式就是去除某一类的组件,然后通过不断自测的方式,找出最有可能影响渲染效率的元素:

SearchSelect(基于antd的Select封装的一个业务组件)。

所以,影响渲染性能的元素很可能就是它。

渲染性能

除了组件的问题,还有可能是渲染的问题。

首先,原来无限滚动的逻辑就是基于scroll事件,通过不断滚动触发的回调,重新计算渲染到页面上的区间。

其次,为了动态调整可视区域的元素,使用了MutationObserver。

导致空白问题则会有这几种可能:

  1. 没加防抖,频繁渲染带来性能消耗
  2. scroll 和 MutationObserver 相继执行了渲染,导致dom出现了跳动的现象。
  3. 预留的元素个数,viewPrepareCount太小了,导致拖动太快时,后面或前面都没有多余的可见元素
  4. 没有开启GPU加速,应该使用transform代替top来定位到正确位置

不幸运的是,以上的可能都一一排除后,发现几乎没有啥提升。

其实,在第二点缩小范围时,应该意识到,空白问题/拖动不流畅均是因为渲染性能低下导致的

测试验证

1. 虚拟列表 rc-virtual-list

为了验证是Select 组件的问题,基于:

rc-virtual-list

做了一个在线 demo :‍‍‍‍

在线地址:https://codesandbox.io/s/optimistic-bartik-69ygc?file=/src/animation.tsx

动态演示:

38642496350c5026777cb61152c016be.gif

这里渲染了1000 条记录, 每条记录里有5个select;

默认使用的是 antd Select, 几乎拉不动;

切换到原生select之后, 如丝般顺滑。

由此可以确定,卡顿是 Slect 组件引起的。

所以要减少渲染成本:

  1. 减少自己的父组件渲染成本,React.memo/React.useMemo/React.useCallback.
  2. 减少Select渲染成本(比较麻烦,而且效果不明显。经过自测,仅仅是使用一个基础的Select,rc无限滚动的情况下同样发生了卡顿)

2. 下拉懒加载

基于 Intersection Observer 实现一个 下拉懒加载。

利用 Intersection Observer 实现:

在列表的底部(也可能是底部偏上的某个位置)插入一个observer-dom元素.

通过Observer来观测其是否在可视区域中,如果在,那么就往下加载更多的内容:

初始状态时,列表会多渲染几条数据(两屏数据),observer-dom元素一直被顶到底部.

用户往下滚动时,observer-dom元素“出现”在用户视野。

每次多加载一屏的数据,循环如此,直到整个列表都渲染到页面上。

在线demo: https://codesandbox.io/s/gundongjiazai-antd491-forked-vtchw?file=/index.js

动态演示:

d1922dcfca1f3c4b1b125ef797b6f11f.gif

选择方案

  1. 要么接受使用rc无限滚动的不够流畅;
  2. 要么使用 Intersection Observer 实现一个下拉懒加载的无限滚动效果

最终采用下拉懒加载。


总结

通常,无限滚动的方案可以分为两种:

1. 虚拟长列表

  • 优点:可以保证渲染在页面上的dom元素尽可能少

  • 缺点:如果没有特殊处理(比如rc或锁定滚动区域),快速滚动时,基本都会有闪动的情况(也就是本次的空白问题)

2. 下拉懒加载

  • 优点:防止用户快速拖动的出现闪动问题。再通过加一个loading效果,帮助优化体验

  • 缺点:当用户把列表拉到底,整个列表都会被渲染到页面上

在选择虚拟长列表or下拉懒加载之间的取舍时,可以参考:

如果闪动问题可以接受(组件渲染没有太大性能问题),而且对dom数量要求很严格,那么选择虚拟长列表会更好。

如果闪动问题不能接受,而最终的dom数量能够接受,那么选择下拉蓝加载会更好。

无论是选择虚拟长列表or下拉懒加载,在使用监听scroll事件或者Intersetion Observer API之间的取舍时,可以参考:

  • scroll的事件回调会在主线程中被成千上万次调用,尽管加了防抖
  • scroll的方式,需要不断记录scrollTop和元素高度

而使用Intersetion Observer API,上述几点的计算就可以省略了,优化工作交给了浏览器。如果不考虑IE 等, 它是一个不错的选择。

内容就这么多, 希望对大家有所启发。

如有错误, 欢迎指正, 谢谢。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/573078.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

2016-5-31 问题及解决

1.执行linux版本时,碰到官方默认发布的cef3二进制包,不支持在动态库中初始化cef3现象:启动时报一下错误->FATAL:content_main_runner.cc(662)] Check failed: base::allocator::IsAllocatorInitialized()原因:在共享库中初始化…

mysql fio测试_Linux下 fio磁盘压测笔记

专业的测试磁盘IO性能的软件:1、fio 【推荐使用】2、iometerfio的使用:参考博客: http://lilinji.blog.51cto.com/5441000/1569623https://linux.die.net/man/1/fiohttp://elf8848.iteye.com/blog/2168876 【讲的比较好】注意,我…

清除浮动塌陷的4种经典套路

【自古深情留不住,总是套路得人心。】 这些日子,各个朋友圈随时可以见到这句话的身影。的确人们常说:“多一点真诚,少一点套路。”,但是最终现象写实也是那么残酷和无奈。 同样在前端攻城狮的道路,我们都是…

mysql有没有开窗函数_mysql实现开窗函数

mysql实现开窗函数开窗分析函数简单来讲就是相似partition by aaa, bbb order by ccc这样的形式mysql在使用mysql的过程当中遇到过这样的需求,须要求出一段时间内每一个item的最大值及对应的那一条记录。最大值很容易使用group by item求出来,可是求出那…

java 格式化时间计算

package com.roshi.utils.common;import java.text.ParseException;import java.text.SimpleDateFormat;import java.util.Calendar;import java.util.Date;public class DateUtilsCom {/*** 格式化时间 * param date* return* author lzq* throws ParseException * date 2016年…

微信小程序点播插件_微信小程序 wxParse插件显示视频问题

修改wxParse/html2json.js 文件 ,在 html2json(html, bindName)方法里var node {node: element,tag: tag,};这里使用 node.tag 能够获取标签类型,比如: img、embed、video这里我的系统后台编辑器使用的是ckeditor 上传视频。生成的便签是&am…

GridView - Batch Editing - How to cancel editing or disable the editor conditionally

Please refer to :https://www.devexpress.com/Support/Center/Example/Details/T115116转载于:https://www.cnblogs.com/michelledawm/p/5980940.html

linkerd mysql_Linkerd和k8s无法正常工作

我试图让我的头在kubernetes的链接。我在我的本地minikube中使用他们网站上的linkerd deamonset示例它全部部署在production命名空间中。当我尝试http_proxy$(kubectl --namespaceproduction get svc l5d -o jsonpath"{.status.loadBalancer.ingress[0].*}"):4140 cu…

mssql 跨库获取mysql

exec sp_addlinkedserver serverMYSQL, --这是链接服务器的名称 srvproductmysql, --这个你自己随便吧 providerMSDASQL, --这是固定的,不能瞎填,否则死定了 datasrcmyserver, --这是ODBC里面data source name,系统DNS locationNULL, provstrDRIVER{MyS…

mysql standalone模式_Boss7 Standalone模式开发配置

PS:以下配置均添加到standalone.xml文件中。1.Get请求编码设置,将如下配置添加到标签之后2.开启开发模式:搜索“urn:jboss:domain:web:1.1”,在此标签下添加如下配置:3.虚拟域名设置搜索“urn:jboss:domain:web:1.1”&…

机器学习笔记——深度学习入门篇

前向传播与反向传播转载于:https://www.cnblogs.com/hdu-cpd/p/5988606.html

java 模板模式_Java模板模式(Template模式)

Java模板模式(Template模式)下一节>Template模式定义:定义一个操作中算法的骨架,将一些步骤的执行延迟到其子类中。其实Java的抽象类本来就是Template模式,因此使用很普遍。而且很容易理解和使用,我们直接以示例开始&#xff1…

python ide: pycharm

1, 设置python路径 2,运行py文件 https://www.jetbrains.com/help/pycharm/2016.1/creating-and-running-your-first-python-project.html#d50296e226 3, 格式化代码 From the main menu, select Code | Reformat Code or press CtrlAltL. 转载于:https://www.cnblo…

pandas入门(2)

使用dataframe的列当做索引 frame DataFrame({a:range(7),b:range(7,0,-1),c:[one,one,one,two,two,two,two],d:[0,1,2,0,1,2,3]}) frame.set_index([c,d]) a b c d one 0 0 7 1 1 6 2 2 5 two 0 3 4 1 4 3 2 5 2 3 6 1 可以使用frame.set_index([c,d],dropFalse)将…

IP地址框

//IP地址框 // 此段代码:独立的获取本机IP地址和计算机名 WORD wVersionRequested; WSADATA wsaData; char name[255]; CString ip; PHOSTENT hostinfo; wVersionRequested MAKEWORD(2, 0); if (WSAStartup(wVersionRequested, &wsaData) 0) { if(ge…

java 释放数组_java集合ArrayList中clear方法内存释放分析

最近在看ArrayList源码的时候看到了ArrayList的clear方法&#xff0c;源码如下&#xff1a;public void clear() {modCount;// clear to let GC do its workfor (int i 0; i < size; i)elementData[i] null;size 0;}在看到这段代码的时候联想到内存释放的问题如下代码&am…

Android 滑动菜单框架--SwipeMenuListView框架完全解析

SwipeMenuListView&#xff08;滑动菜单&#xff09; A swipe menu for ListView.--一个非常好的滑动菜单开源项目。 Demo 一、简介 看了挺长时间的自定义View和事件分发&#xff0c;想找一个项目练习下。。正好印证自己所学。 在github上找到了这个项目&#xff1a;SwipeMenuL…

【Web网站服务器开发】Apache 和 Tomcat的区别及配置

Apache 和 Tomcat 都是web网络服务器&#xff0c;两者既有联系又有区别&#xff0c;在进行HTML、PHP、JSP、Perl等开发过程中&#xff0c;需要准确掌握其各自特点&#xff0c;选择最佳的服务器配置。 apache是web服务器&#xff08;静态解析&#xff0c;如HTML&#xff09;&…

java字符生成器_Java实现简单字符生成器代码例子

创建成功的字符串对象&#xff0c;其长度是固定的&#xff0c;内容是不能被修改和编辑。虽然使用""可以达到增加新字符或字符串的目的&#xff0c;但""会产生一个新的String实例,会在内存中创建新的字符串对象。如果重复地对字符串进行修改&#xff0c;将极…

ES基础知识

马克。。。转载于:https://www.cnblogs.com/lirunzhou/p/6012321.html