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

最近在做一个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...

参考

相关推荐
moshuying1 小时前
别让AI焦虑,偷走你本该有的底气
前端·人工智能
GIS之路2 小时前
ArcPy,一个基于 Python 的 GIS 开发库简介
前端
可夫小子3 小时前
OpenClaw基础-为什么会有两个端口
前端
喝拿铁写前端4 小时前
Dify 构建 FE 工作流:前端团队可复用 AI 工作流实战
前端·人工智能
喝咖啡的女孩4 小时前
React 合成事件系统
前端
从文处安5 小时前
「九九八十一难」组合式函数到底有什么用?
前端·vue.js
用户5962585736065 小时前
戴上AI眼镜逛花市——感受不一样的体验
前端
yuki_uix5 小时前
Props、Context、EventBus、状态管理:组件通信方案选择指南
前端·javascript·react.js
老板我改不动了5 小时前
前端面试复习指南【代码演示多多版】之——HTML
前端
panshihao5 小时前
Mac 环境下通过 SSH 操作服务器,完成前端静态资源备份与更新(全程实操无坑)
前端