「虚拟滚动」也没那么难嘛

最近在做一个React Native项目:需要滑动展示成千上万的图片。直接堆上去性能爆炸,又没有好用的虚拟滚动基建,于是自己动手,造一个!(为了方便展示,写了一个 web 版的 demo)

在线体验:leigithub1024.github.io/VirtualScro...

仓库地址:github.com/LeiGitHub10...

老方案的问题

这个项目以前只需要展示几百张图片,所以不需要做什么优化,一次请求好,直接堆上去就好了。但新版本需要展示的数据可能成千上万,如果一次请求全量数据,需要几分钟,滑动起来也会很卡。

尽管 React Native 提供了如FlatList、SectionList等虚拟滚动组件,但它们不完全符合复杂的业务需求且灵活性不足。因此,我决定自己动手,丰衣足食!

核心功能 - 动态加载

只渲染可见部分这是虚拟滚动的核心思想,用到谁就请求谁,渲染谁。其他一切工作都是围绕这一点出发的。

一种简单的解决方案是给每个元素添加一个监听器,如果在屏幕范围内就展示,不在就隐藏。但这样过多的监听器也会带来大量开销。更经济的做法是只监听父容器,根据父容器的当前位置,计算哪些需要展示,哪些不需要。代码如下:

js 复制代码
    //计算左右边界
    const container = document.querySelector('.slide-container');
    const scrollLeft = container.scrollLeft;      
    const containerWidth = container.clientWidth;
    const firstVisibleIndex = Math.floor(scrollLeft / 100 ) ; // 每个项的宽度是 100px
    const lastVisibleIndex = Math.ceil((scrollLeft + containerWidth) / 100) ;
    //挨个判断需不需要展示
    document.querySelectorAll('.item-placeholder').forEach((element, index) => {
        if (index >= firstVisibleIndex && index <= lastVisibleIndex) {
          if (!element.querySelector('img')) { // 检查是否已加载图片
            let img = document.createElement('img');
            img.src = `https://s11.ax1x.com/2023/01/30/pSdWF7F.png`; 
            img.style.height = "100%";
            img.style.objectFit = "cover";
            element.appendChild(img);
          }
          element.classList.add('item-active');
        } else {
          element.classList.remove('item-active');
          let img = element.querySelector('img');
          if (img) { // 如果存在图片则移除,防止占用过多内存
            element.removeChild(img);
          }
        }
      });

优化 - 节流

滑动是很灵敏的事件,为了进一步提高性能,需要加上节流机制。同时为了保证滑动停止时数据能够及时更新,需要监听滑动停止事件,并额外触发一次更新。

js 复制代码
   const handleScroll = throttle(function() {
      freshData();
      if (scrollTimeout !== null) {clearTimeout(scrollTimeout)}
      scrollTimeout = setTimeout(handleScrollEnd, 500); // 500ms后无新的滑动事件,认为滑动结束
    }, 250); // 250ms 的节流,防止滚动时频繁触发

优化 - 缓冲区

简单地扩大缓冲区,可以显著改善用户的滚动体验,减少加载的等待时间。

js 复制代码
   const buffer = 20; // 增加20个元素的 buffer
   const firstVisibleIndex = Math.floor(scrollLeft / 100 ) - buffer; // 每个项的宽度是 100px
   const lastVisibleIndex = Math.ceil((scrollLeft + containerWidth) / 100) + buffer;

demo 效果展示

做完上述工作,我们便掌握了虚拟滚动最关键的部分。我用 M1 芯片的 Mac 测试,10 万条以内的数据很流畅,即使有 100 万数据,也不至于卡死。

在线体验:leigithub1024.github.io/VirtualScro...

仓库地址:github.com/LeiGitHub10...

参考

相关推荐
一个处女座的程序猿O(∩_∩)O2 小时前
小型 Vue 项目,该不该用 Pinia 、Vuex呢?
前端·javascript·vue.js
hackeroink5 小时前
【2024版】最新推荐好用的XSS漏洞扫描利用工具_xss扫描工具
前端·xss
迷雾漫步者6 小时前
Flutter组件————FloatingActionButton
前端·flutter·dart
向前看-7 小时前
验证码机制
前端·后端
燃先生._.8 小时前
Day-03 Vue(生命周期、生命周期钩子八个函数、工程化开发和脚手架、组件化开发、根组件、局部注册和全局注册的步骤)
前端·javascript·vue.js
高山我梦口香糖9 小时前
[react]searchParams转普通对象
开发语言·前端·javascript
m0_748235249 小时前
前端实现获取后端返回的文件流并下载
前端·状态模式
m0_7482402510 小时前
前端如何检测用户登录状态是否过期
前端
black^sugar10 小时前
纯前端实现更新检测
开发语言·前端·javascript