分段渲染加载页面

前言

在日常工作中我们一定会遇到某个页面需要渲染比较多的内容,比如列表或者文字;一般当数据过多的时候,就会出现页面加载比较慢,很长时间没有内容的展示,用户体验不太好,当然这时候就有人会说直接使用虚拟滚动嘛,只展示其中一部分数据。这种确实是一个非常好的办法,但是如果页面可能不能使用虚拟滚动实现呢(虽然少,但是有可能),此时我们就需要实现分段渲染

一、分段渲染是什么?

分段渲染就是每次只渲染一部分数据,比如我有1w条数据,我每次只渲染其中10条或者20条数据。

二、实现

1.不使用分段渲染

首先我们先看一下不使用分段渲染,直接渲染全部数据是什么样子的,我们先降低浏览器的CPU,模拟低配置的运行方式

这里我们使用推荐的

javascript 复制代码
<template>
  <div>
    <ul>
      <li v-for="item in total" :key="item.id" style="margin-top: 10px">
        <div>显示其他内容</div>
        {{ item.name }}
      </li>
    </ul>
  </div>
</template>
<script setup lang="ts">
import { ref } from "vue";
const observer = new PerformanceObserver((list) => {
  console.log(list);
  const entries = list.getEntriesByName("first-paint");
  if (entries.length > 0) {
    const fcpTime = entries[0].startTime;
    console.log("First Contentful Paint:", fcpTime, "ms");
    observer.disconnect(); // 获取到数据后断开观察器
  }
});

observer.observe({ type: "paint", buffered: true });
const total = ref([]);
// 造1000条数据
const startTime = performance.now();
for (let i = 0; i < 10000; i++) {
  total.value.push({
    id: i,
    name: `name${i}`,
    age: 18 + i,
  });
}
const endTime = performance.now();
console.log(`数据生成,耗时:${endTime - startTime}ms`);


我们能看到,FCP的时间高达1000+ms,每次刷新有偏差,但是基本都是1000+的

2 使用分段渲染

javascript 复制代码
<template>
  <div>
    <ul>
      <li v-for="item in list" :key="item.id" style="margin-top: 10px">
        <div>显示其他内容</div>
        {{ item.name }}
      </li>
    </ul>
  </div>
</template>
<script setup lang="ts">
import { ref } from "vue";
const observer = new PerformanceObserver((list) => {
  const entries = list.getEntriesByName("first-paint");
  if (entries.length > 0) {
    const fcpTime = entries[0].startTime;
    console.log("First Contentful Paint:", fcpTime, "ms");
    observer.disconnect(); // 获取到数据后断开观察器
  }
});

observer.observe({ type: "paint", buffered: true });
const total = ref([]);
// 造1000条数据
const startTime = performance.now();
for (let i = 0; i < 10000; i++) {
  total.value.push({
    id: i,
    name: `name${i}`,
    age: 18 + i,
  });
}
const endTime = performance.now();
console.log(`数据生成,耗时:${endTime - startTime}ms`);
const list = ref([]);
// 使用requestAnimationFrame分批次渲染
const renderBatch = () => {
  const batchSize = 100;
  const totalItems = total.value.length;
  let startIndex = 0;
  const renderNextBatch = () => {
    const endIndex = startIndex + batchSize;
    const batchItems = total.value.slice(startIndex, endIndex);

    list.value.push(...batchItems);
    startIndex = endIndex;

    if (startIndex < totalItems) {
      requestAnimationFrame(renderNextBatch);
    }
  };
  renderNextBatch();
};
renderBatch();

这里使用requestAnimationFrame进行渲染,如果不清楚的可自行查阅mdn文档

此时我们看看结果


此时结果最大1088ms,也能看出来提升较为明显;

不过会出现滚动条会一直变小,因为不断在渲染加载新的数据,当然这也不算什么大问题;

当然也不一定非得用requestAnimationFrame,用延时器也是一样的,目的都是为了每次只加载一部分数据

总结

以上就是关于通过使用分段渲染来达到快速加载数据的方法,当然,这种方式并不是最好的,因为真正涉及到大数据列表还是虚拟滚动的性能和效果最好。

相关推荐
吃饺子不吃馅10 分钟前
如何设计一个 Canvas 事件系统?
前端·canvas·图形学
Baklib梅梅25 分钟前
无头内容管理系统:打造灵活高效的多渠道内容架构
前端·ruby on rails·前端框架·ruby
over6971 小时前
浏览器里的AI魔法:用JavaScript玩转自然语言处理
前端·javascript
渣渣盟1 小时前
探索Word2Vec:从文本向量化到中文语料处理
前端·javascript·python·文本向量化
Pu_Nine_91 小时前
Vue 3 + TypeScript 项目性能优化全链路实战:从 2.1MB 到 130KB 的蜕变
前端·vue.js·性能优化·typescript·1024程序员节
云枫晖2 小时前
Webpack系列-Loader
前端·webpack
aggression2 小时前
代码敲击乐:让你了解前端的动静结合和移动端的适配性
前端
yinuo2 小时前
深入理解与实战 Git Submodule
前端
骑自行车的码农2 小时前
React 事件收集函数
前端·react.js
一个处女座的程序猿O(∩_∩)O2 小时前
Vue CLI 插件开发完全指南:从原理到实战
前端·javascript·vue.js