一场“数据海啸”,让我重新认识了 requestAnimationFrame

那是一个普通的下午,我正悠哉地喝着咖啡,准备轻松完成一个"列表渲染"的小需求。后端同事发来消息:

"小王,接口写好了,就几条数据,前端渲染一下就行。"

我点开接口一看,整个人差点从椅子上滑下来:
10 万条数据!

脑子里立刻浮现出一种画面:就像港口突然迎来一艘超级货船,把几十万件货物一股脑儿砸到我面前,而我却只有一辆小推车去搬运。

但我还是硬着头皮上了,写下最直观的循环渲染:

js 复制代码
const container = document.querySelector(".list-container");
const dataList = new Array(100000).fill('我是数据项').map((t, i) => `${t} - ${i + 1}`);

for (let i = 0; i < dataList.length; i++) {
  const div = document.createElement('div');
  div.innerText = dataList[i];
  container.appendChild(div);
}

结果毫不意外:

  • 浏览器瞬间卡死,鼠标动不了
  • 页面白屏,用户完全无法操作
  • 内存一路飙升,Chrome 差点崩溃

这就像把全部货物直接砸在港口,结果不仅搬运停滞,还把整个码头压塌了。

1.为什么会这样?

  • JS 执行阻塞:创建 10 万个 DOM 节点让主线程无暇他顾
  • 渲染延迟:浏览器还没来得及"喘气",页面就被压垮
  • 内存压力:DOM 堆积过多,机器性能直接跪了

我盯着屏幕发呆,心想:

"难道就没有一种方法,可以像分批卸货一样,把这些数据一点点渲染到页面里?"

2.关键时刻的救星:requestAnimationFrame

就在这时,我想起一个被我长期忽视的 API:requestAnimationFrame

它的特性正好符合我的需求:

  • 浏览器会在下一帧渲染前 调用回调(大约 16.6ms 一次)
  • 能和刷新频率保持一致
  • 不会长时间霸占主线程

于是,我决定换一种思路:分批渲染

js 复制代码
function renderBigData(data, chunkSize = 50) {
  const container = document.querySelector(".list-container");
  let index = 0;

  function renderChunk() {
    const chunkEnd = Math.min(index + chunkSize, data.length);
    const fragment = document.createDocumentFragment();

    for (; index < chunkEnd; index++) {
      const div = document.createElement('div');
      div.innerText = data[index];
      fragment.appendChild(div);
    }

    container.appendChild(fragment);

    if (index < data.length) {
      requestAnimationFrame(renderChunk); // 下一帧继续
    }
  }

  renderChunk();
}

// 测试渲染 1 万条数据
const dataList = new Array(10000).fill('我是数据项').map((t, i) => `${t} - ${i + 1}`);
renderBigData(dataList);

效果瞬间不一样了:

✅ 页面不卡顿,每次只渲染 50 条,主线程得以喘息

✅ 用户依然能点击、滚动、输入

✅ 渲染过程像流水一样自然,不再"一刀切"


3.进阶优化:虚拟滚动(Virtual Scroll)

然而,问题并没有结束。

如果数据量达到 几十万甚至上百万条,哪怕分批渲染,DOM 仍然会堆积如山。

这时,真正的"性能核武器"登场了:虚拟滚动

原理就像"橱窗展示"

  • 商场仓库里有 10 万件衣服,但橱窗只摆几十件
  • 用户走过来时,再换上他们能看到的那一批

简单代码示例:

js 复制代码
function renderVirtualScroll(data, container, itemHeight = 30) {
  const viewportHeight = container.clientHeight;
  const visibleCount = Math.ceil(viewportHeight / itemHeight);
  let startIndex = 0;

  function updateVisibleItems() {
    const endIndex = startIndex + visibleCount;
    const visibleData = data.slice(startIndex, endIndex);

    container.innerHTML = '';
    visibleData.forEach(item => {
      const div = document.createElement('div');
      div.style.height = `${itemHeight}px`;
      div.textContent = item;
      container.appendChild(div);
    });
  }

  container.addEventListener('scroll', () => {
    startIndex = Math.floor(container.scrollTop / itemHeight);
    updateVisibleItems();
  });

  updateVisibleItems();
}

这样,无论数据量有多大,DOM 中真正存在的始终只有几十个元素,性能丝滑无比。


4.那么,到底该选哪种方案?

方案 适用场景 优点 缺点
直接渲染 < 5k 条数据 简单粗暴 数据量一大就爆炸
分批渲染 (rAF) 5k ~ 10w 条 页面不卡顿,实现简单 DOM 节点仍很多
虚拟滚动 10w+ 条 性能极致,内存占用低 实现较复杂

5.总结

前端性能优化,不只是让代码能跑,而是让用户感到"丝滑"。

如果把数据渲染比作一场货物运输:

  • 直接渲染是一次性压港口
  • 分批渲染是分车次慢慢运
  • 虚拟滚动则是橱窗展示,给你看你需要的

requestAnimationFrame,就是那个让运输过程变得优雅、从容的调度员; 所以下次,当你也遇到"数据海啸"时,交给它吧。

相关推荐
Evan Wang1 小时前
深度解析GetX依赖注入,从Spring与Vue视角看Flutter架构
vue.js·spring boot·flutter
veneno8 小时前
大量异步并发请求控制并发解决方案
前端
i***t9198 小时前
Spring Boot项目接收前端参数的11种方式
前端·spring boot·后端
oden8 小时前
2025博客框架选择指南:Hugo、Astro、Hexo该选哪个?
前端·html
小光学长9 小时前
基于ssm的宠物交易系统的设计与实现850mb48h(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。
java·前端·数据库
云中飞鸿9 小时前
函数:委托
javascript
小小前端要继续努力9 小时前
渐进增强、优雅降级及现代Web开发技术详解
前端
老前端的功夫10 小时前
前端技术选型的理性之道:构建可量化的ROI评估模型
前端·javascript·人工智能·ubuntu·前端框架
汝生淮南吾在北10 小时前
SpringBoot+Vue超市收银管理系统
vue.js·spring boot·后端
狮子座的男孩10 小时前
js函数高级:04、详解执行上下文与执行上下文栈(变量提升与函数提升、执行上下文、执行上下文栈)及相关面试题
前端·javascript·经验分享·变量提升与函数提升·执行上下文·执行上下文栈·相关面试题