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