requestIdleCallback解决页面卡顿

一、基本概念

  1. js是单线程执行的,分为GUI渲染线程,js引擎线程,事件触发线程,定时器触发器线程,http请求线程。
  2. 执行js会阻塞gui渲染,如js执行所需时间超过一帧的时间,就会导致页面掉帧,也就是肉眼看到的卡顿。

二、场景模拟

1. 准备工作

通过一个小方块平移运动模拟浏览器渲染图形

css 复制代码
.block {
    position: absolute;
    left: 100px;
    top: 100px;
    width: 50px;
    height: 50px;
    background: #f00;
    animation: move 3s linear infinite alternate;
}

@keyframes move {
    0% {
        left: 100px;
    }
    100% {
        left: 300px;
    }
}

使用while模拟js阻塞

js 复制代码
function task() {
    const now = performance.now()
    while(performance.now() - now < 1) {}
}
2. 测试正常运行js耗费时间
js 复制代码
function run1() {
    const data = new Array(2000)
    console.time("normal")
    for(let i = 0; i < data.length; i++) {
        task()
    }
    console.timeEnd("normal")
}

可以看到总耗时2s,页面在js运行期间卡住了,小方块完全没有响应,直到js运行结束后才正常播放动画。

3. 测试使用宏任务方案延迟计算
js 复制代码
function setTimeoutTask() {
    return new Promise((s) => {
        setTimeout(() => {
            task()
            s()
        }, 0)
    }) 
}
function run2() {
    const data = new Array(2000)
    console.time("hong")
    let list = []
    for(let i = 0; i < data.length; i++) {
        list.push(setTimeoutTask())
    }
    Promise.all(list).then(() => {
        console.timeEnd("hong")
    })
}

宏任务方案基本原理是利用setTimeout事件会放到下一个宏任务内运行,从而减少了单个渲染帧的计算压力,但还是会出现间断卡顿的现象。

4. 测试使用空闲帧方案延迟计算
js 复制代码
function idleTask(task) {
    return new Promise((s) => {
        window.requestIdleCallback((IdleDeadline) => {
            let timeRemain = IdleDeadline.timeRemaining();
            if(timeRemain > 0) {
                task()
                s()
            } else {
                idleTask(task).then(() => {
                    s()
                })
            }
        })
    })
}
function run3() {
    const data = new Array(2000)
    console.time("idle")
    let list = []
    for(let i = 0; i < data.length; i++) {
        list.push(idleTask(task))
    }
    Promise.all(list).then(() => {
        console.timeEnd("idle")
    })
}

这里用到requestIdleCallback API,MDN上的说明是"这个函数将在浏览器空闲时期被调用。这使开发者能够在主事件循环上执行后台和低优先级工作,而不会影响延迟关键事件,如动画和输入响应"。个人理解就是在每一帧结束后,如果有空闲时间就会调用。

函数回调内有个值"IdleDeadline.timeRemaining()"用于说明当前帧还剩余多少时间。如果大于0,说明有剩余时间可以用于执行js计算。

实际效果确实是没有卡顿感了,但是相应的执行耗时也比普通执行要长

demo: code.juejin.cn/api/raw/733...

相关推荐
fishmemory7sec7 分钟前
Electron 主进程与渲染进程、预加载preload.js
前端·javascript·electron
fishmemory7sec10 分钟前
Electron 使⽤ electron-builder 打包应用
前端·javascript·electron
豆豆1 小时前
为什么用PageAdmin CMS建设网站?
服务器·开发语言·前端·php·软件构建
JUNAI_Strive_ving1 小时前
番茄小说逆向爬取
javascript·python
看到请催我学习2 小时前
如何实现两个标签页之间的通信
javascript·css·typescript·node.js·html5
twins35202 小时前
解决Vue应用中遇到路由刷新后出现 404 错误
前端·javascript·vue.js
qiyi.sky2 小时前
JavaWeb——Vue组件库Element(3/6):常见组件:Dialog对话框、Form表单(介绍、使用、实际效果)
前端·javascript·vue.js
煸橙干儿~~2 小时前
分析JS Crash(进程崩溃)
java·前端·javascript
哪 吒2 小时前
华为OD机试 - 几何平均值最大子数(Python/JS/C/C++ 2024 E卷 200分)
javascript·python·华为od
安冬的码畜日常2 小时前
【D3.js in Action 3 精译_027】3.4 让 D3 数据适应屏幕(下)—— D3 分段比例尺的用法
前端·javascript·信息可视化·数据可视化·d3.js·d3比例尺·分段比例尺