requestIdleCallback 和 requestAnimationFrame
页面流畅与 FPS
页面是一帧一帧绘制出来的,当每秒绘制的帧数(FPS)达到 60 时,页面是流畅的,小于这个值时,用户会感觉到卡顿。
1s 60 帧,所以每一帧分到的时间是 1000/60 ≈ 16 ms。所以我们书写代码时力求不让一帧的工作量超过 16ms。
Frame
那么浏览器每一帧都需要完成哪些工作?
通过上图可看到,一帧内需要完成如下六个步骤的任务:
- 处理用户的交互
- JS 解析执行
- 帧开始。窗口尺寸变更,页面滚去等的处理
- requestAnimationFrame(rAF)
- 布局
- 绘制
或者说
一帧内做了什么事呢
1.处理用户的事件,就是 event 例如 click,input change 等。
2.执行定时器任务
3.执行 requestAnimationFrame
4.执行 dom 的回流与重绘
5.计算更新图层的绘制指令
6.绘制指令合并主线程 如果有空余时间会执行 requestidlecallback
requestIdleCallback
上面六个步骤完成后没超过 16 ms,说明时间有富余,此时就会执行 requestIdleCallback
里注册的任务。
从上图也可看出,和 requestAnimationFrame
每一帧必定会执行不同,requestIdleCallback
是捡浏览器空闲来执行任务。
如此一来,假如浏览器一直处于非常忙碌的状态,requestIdleCallback
注册的任务有可能永远不会执行。此时可通过设置 timeout
(见下面 API 介绍)来保证执行。
API
var handle = window.requestIdleCallback(callback[, options])
callback
:回调,即空闲时需要执行的任务,该回调函数接收一个
IdleDeadline
对象作为入参。其中
IdleDeadline
对象包含:
didTimeout
,布尔值,表示任务是否超时,结合timeRemaining
使用。timeRemaining()
,表示当前帧剩余的时间,也可理解为留给任务的时间还有多少。
options
:目前 options 只有一个参数
timeout
。表示超过这个时间后,如果任务还没执行,则强制执行,不必等待空闲。
IdleDeadline
对象参考 MDN:developer.mozilla.org/zh-CN/docs/…
示例
requestIdleCallback(myNonEssentialWork, { timeout: 2000 });
// 任务队列
const tasks = [
() => {
console.log("第一个任务");
},
() => {
console.log("第二个任务");
},
() => {
console.log("第三个任务");
},
];
function myNonEssentialWork (deadline) {
// 如果帧内有富余的时间,或者超时
while ((deadline.timeRemaining() > 0 || deadline.didTimeout) && tasks.length > 0) {
work();
}
if (tasks.length > 0)
requestIdleCallback(myNonEssentialWork);
}
function work () {
tasks.shift()();
console.log('执行任务');
}
超时的情况,其实就是浏览器很忙,没有空闲时间,此时会等待指定的 timeout
那么久再执行,通过入参 dealine
拿到的 didTmieout
会为 true
,同时 timeRemaining ()
返回的也是 0。超时的情况下如果选择继续执行的话,肯定会出现卡顿的,因为必然会将一帧的时间拉长。
cancelIdleCallback
与 setTimeout
类似,返回一个唯一 id,可通过 cancelIdleCallback
来取消任务。
总结
一些低优先级的任务可使用 requestIdleCallback
等浏览器不忙的时候来执行,同时因为时间有限,它所执行的任务应该尽量是能够量化,细分的微任务(micro task)。
因为它发生在一帧的最后,此时页面布局已经完成,所以不建议在 requestIdleCallback
里再操作 DOM,这样会导致页面再次重绘。DOM 操作建议在 rAF 中进行。同时,操作 DOM 所需要的耗时是不确定的,因为会导致重新计算布局和视图的绘制,所以这类操作不具备可预测性。
Promise 也不建议在这里面进行,因为 Promise 的回调属性 Event loop 中优先级较高的一种微任务,会在 requestIdleCallback
结束时立即执行,不管此时是否还有富余的时间,这样有很大可能会让一帧超过 16 ms。
额外补充一下window.requestAnimationFrame
在没有 requestAnimationFrame
方法的时候,执行动画,我们可能使用 setTimeout
或 setInterval
来触发视觉变化;但是这种做法的问题是:回调函数执行的时间是不固定的,可能刚好就在末尾,或者直接就不执行了,经常会引起丢帧而导致页面卡顿。
归根到底发生上面这个问题的原因在于时机,也就是浏览器要知道何时对回调函数进行响应。setTimeout
或 setInterval
是使用定时器来触发回调函数的,而定时器并无法保证能够准确无误的执行,有许多因素会影响它的运行时机,比如说:当有同步代码执行时,会先等同步代码执行完毕,异步队列中没有其他任务,才会轮到自己执行。并且,我们知道每一次重新渲染的最佳时间大约是 16.6 ms,如果定时器的时间间隔过短,就会造成 过度渲染,增加开销;过长又会延迟渲染,使动画不流畅。
requestAnimationFrame
方法不同与 setTimeout
或 setInterval
,它是由系统来决定回调函数的执行时机的,会请求浏览器在下一次重新渲染之前执行回调函数。无论设备的刷新率是多少,requestAnimationFrame
的时间间隔都会紧跟屏幕刷新一次所需要的时间;例如某一设备的刷新率是 75 Hz,那这时的时间间隔就是 13.3 ms(1 秒 / 75 次)。需要注意的是这个方法虽然能够保证回调函数在每一帧内只渲染一次,但是如果这一帧有太多任务执行,还是会造成卡顿的;因此它只能保证重新渲染的时间间隔最短是屏幕的刷新时间。
requestAnimationFrame
方法的具体说明可以看 MDN 的相关文档,下面通过一个网页动画的示例来了解一下如何使用。
复制代码let offsetTop = 0;
const div = document.querySelector(".div");
const run = () => {
div.style.transform = `translate3d(0, ${offsetTop += 10}px, 0)`;
window.requestAnimationFrame(run);
};
run();
如果想要实现动画效果,每一次执行回调函数,必须要再次调用 requestAnimationFrame
方法;与 setTimeout
实现动画效果的方式是一样的,只不过不需要设置时间间隔。
参考:https://juejin.cn/post/6844903848981577735
这个跟react
的 fiber
的有什么关系?
因为 react 也有该机制 但是 react 并没有用 requestidlecallback
,说是这个东西经过测试可能会超过 16ms,超过 16ms 绘制就会看起来很卡 所以 react16 是用的 requestAnimationFrame + postMessage
实现的
那为什么不用setTimeOut`` setTimeOut
设置为 0 也会有一个最小毫秒延迟 4ms,所以是用了postMessage
,react18 又换成了MessageChannel
实现了队列方式去执行任务。