摘要:二事件循環(huán)與幀事件循環(huán)和上面?zhèn)€名詞的基本概念在此不再啰嗦了,我們著重看下它們之間的關(guān)系。瀏覽器是一個(gè)系統(tǒng),所有的操作最終都會(huì)以頁面的形式展現(xiàn),而頁面的基本單位是幀。當(dāng)某一幀的任務(wù)占用大量時(shí)間的時(shí)候,會(huì)影響到下一幀的執(zhí)行。
歡迎關(guān)注我的公眾號(hào)睿Talk,獲取我最新的文章:
Promise, setTimeout, requestAnimationFrame, requestIdleCallback 這幾個(gè)概念相信很多人都很熟悉了,最近在看 React Fiber 源碼的時(shí)候又對(duì)它們有了更深一層的認(rèn)識(shí),在此分享一下。下文將用 rAF 代表 requestAnimationFrame, rIC 代表 requestIdleCallback。
二、事件循環(huán)與幀事件循環(huán)和上面 4 個(gè)名詞的基本概念在此不再啰嗦了,我們著重看下它們之間的關(guān)系。瀏覽器是一個(gè) UI 系統(tǒng),所有的操作最終都會(huì)以頁面的形式展現(xiàn),而頁面的基本單位是幀。一幀中可能包括的任務(wù)有下面幾種類型。
events: 點(diǎn)擊事件、鍵盤事件、滾動(dòng)事件等
macro: 宏任務(wù),如 setTimeout
micro: 微任務(wù),如 Promise
rAF: requestAnimationFrame
Layout: CSS 計(jì)算,頁面布局
Paint: 頁面繪制
rIC: requestIdleCallback
理想情況下,頁面會(huì)以 60 幀每秒的幀率來運(yùn)行,但實(shí)際上每秒繪制多少幀是由多個(gè)因素決定的,下面舉一些例子:
一個(gè)加載完成的靜態(tài)頁面,當(dāng)用戶沒有進(jìn)行交互的情況下,頁面不需要重繪,幀率為 0。
快速滾動(dòng)頁面的時(shí)候,可視區(qū)域的內(nèi)容不斷發(fā)生變化,瀏覽器會(huì)盡可能快的重繪頁面,理想幀率為 60。
假設(shè)頁面有一個(gè)注冊(cè)了回調(diào)的按鈕,回調(diào)執(zhí)行需要 500 毫秒。當(dāng)點(diǎn)擊按鈕后再快速滾動(dòng)頁面,頭 500 毫秒頁面是卡住動(dòng)不了的,后 500 毫秒會(huì)盡可能快的重繪頁面,這時(shí)候理想幀率為 30。
當(dāng)使用 rAF 制作動(dòng)畫的時(shí)候,瀏覽器會(huì)盡可能快的重繪頁面,桌面瀏覽器可能是 60 幀,移動(dòng)瀏覽器可能是 30 幀。
從上面的例子可以看出,頁面的幀率不是固定的,是會(huì)動(dòng)態(tài)變化的。當(dāng)某一幀的任務(wù)占用大量時(shí)間的時(shí)候,會(huì)影響到下一幀的執(zhí)行。那么誰來調(diào)節(jié)幀率呢?顯然只能依靠瀏覽器自身。作為開發(fā)者的我們是無法準(zhǔn)確預(yù)知回調(diào)什么時(shí)候執(zhí)行的。比如:
function animation() { console.log("time: ", +new Date()); setTimeout(animate, 1000 / 60); } animation();
上面的函數(shù)假定了瀏覽器以幀率 60 來運(yùn)行,但當(dāng)幀率達(dá)不到的時(shí)候,2 幀之間回調(diào)可能執(zhí)行了多次,也可能一次都不執(zhí)行,簡(jiǎn)稱掉幀。
所以在制作動(dòng)畫的時(shí)候,我們不能預(yù)設(shè)瀏覽器的幀率,正確的做法是通過 rAF 注冊(cè)回調(diào), 由瀏覽器來控制動(dòng)畫調(diào)用時(shí)機(jī):
function animation() { console.log("time: ", +new Date()); requestAnimationFrame(animation); } animation();
rAF 會(huì)保證注冊(cè)的回調(diào)在下次渲染頁面之前執(zhí)行,且只會(huì)執(zhí)行一次。另外,當(dāng)頁面處于不可見狀態(tài)時(shí),rAF 會(huì)自動(dòng)停止執(zhí)行,以節(jié)省系統(tǒng)資源。
三、執(zhí)行順序Promise, setTimeout , rAF 和 rIC 對(duì)應(yīng) 4 種隊(duì)列:微任務(wù)隊(duì)列、宏任務(wù)隊(duì)列、animation 隊(duì)列和 idle 隊(duì)列。
微任務(wù)隊(duì)列會(huì)在 JS 運(yùn)行棧為空的時(shí)候立即執(zhí)行。
animation 隊(duì)列會(huì)在頁面渲染前執(zhí)行。
宏任務(wù)隊(duì)列優(yōu)先級(jí)低于微任務(wù)隊(duì)列,一般也會(huì)比 animation 隊(duì)列優(yōu)先級(jí)低,但不是絕對(duì) 。
idle 隊(duì)列優(yōu)先級(jí)最低,當(dāng)瀏覽器有空閑時(shí)間的時(shí)候才會(huì)執(zhí)行。
setTimeout(()=>console.log("setTimeout"), 0); Promise.resolve().then(()=>console.log("promise")); requestAnimationFrame(()=>console.log("animation")); requestIdleCallback(()=>console.log("idle")); // 執(zhí)行結(jié)果大多數(shù)情況下是: promise, animation, setTimeout, idle // 少數(shù)情況是:promise, setTimeout, animation, idle
再來談?wù)効臻e時(shí)間怎么理解。假設(shè)在 1 秒內(nèi)有 3 幀需要渲染:
第一幀,由于宏任務(wù)占用了大量的時(shí)間,沒有空閑時(shí)間。
第二幀,rAF占用的時(shí)間不多,有大量的空閑時(shí)間
第三幀,瀏覽器事件占用的時(shí)間不多,有大量的空閑時(shí)間
與rAF類似,rIC 的執(zhí)行時(shí)機(jī)是由瀏覽器控制的,能更好的保證體驗(yàn),優(yōu)化性能。一般優(yōu)先級(jí)高的任務(wù)(如 UI 更新)會(huì)放在 rAF 隊(duì)列,優(yōu)先級(jí)低的任務(wù)(如日志上傳)會(huì)放 rIC。
四、隊(duì)列特性在一個(gè)事件循環(huán)內(nèi),各個(gè)隊(duì)列有以下特性:
宏任務(wù)隊(duì)列,每次只會(huì)執(zhí)行隊(duì)列內(nèi)的一個(gè)任務(wù)。
微任務(wù)隊(duì)列,每次會(huì)執(zhí)行隊(duì)列里的全部任務(wù)。假設(shè)微任務(wù)隊(duì)列內(nèi)有 100 個(gè) Promise,它們會(huì)一次過全部執(zhí)行完。這種情況下極有可能會(huì)導(dǎo)致頁面卡頓。如果在微任務(wù)執(zhí)行過程中繼續(xù)往微任務(wù)隊(duì)列中添加任務(wù),新添加的任務(wù)也會(huì)在當(dāng)前事件循環(huán)中執(zhí)行,很容易造成死循環(huán), 如:
function loop() { Promise.resolve().then(loop); } loop();
animation 隊(duì)列,跟微任務(wù)隊(duì)列有點(diǎn)相似,每次會(huì)執(zhí)行隊(duì)列里的全部任務(wù)。但如果在執(zhí)行過程中往隊(duì)列中添加新的任務(wù),新的任務(wù)不會(huì)在當(dāng)前事件循環(huán)中執(zhí)行,而是在下次事件循環(huán)中執(zhí)行。
idle 隊(duì)列,每次只會(huì)執(zhí)行一個(gè)任務(wù)。任務(wù)完成后會(huì)檢查是否還有空閑時(shí)間,有的話會(huì)繼續(xù)執(zhí)行下一個(gè)任務(wù),沒有則等到下次有空閑時(shí)間再執(zhí)行。需要注意的是此隊(duì)列中的任務(wù)也有可能阻塞頁面,當(dāng)空閑時(shí)間用完后任務(wù)不會(huì)主動(dòng)退出。如果任務(wù)占用時(shí)間較長(zhǎng),一般會(huì)將任務(wù)拆分成多個(gè)階段,執(zhí)行完一個(gè)階段后檢查還有沒有空閑時(shí)間,有則繼續(xù),無則注冊(cè)一個(gè)新的 idle 隊(duì)列任務(wù),然后退出當(dāng)前任務(wù)。React Fiber 就是用這個(gè)機(jī)制。但最新版的 React Fiber 已經(jīng)不用 rIC 了,因?yàn)檎{(diào)用的頻率太低,改用 rAF 了
五、總結(jié)本文介紹了 4 種隊(duì)列的執(zhí)行順序和每個(gè)隊(duì)列的特性,它們是:宏任務(wù)隊(duì)列、微任務(wù)隊(duì)列、animation 隊(duì)列和 idle 隊(duì)列。實(shí)際應(yīng)用時(shí)可以根據(jù)它們各自的特點(diǎn)分配不同的任務(wù)。
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/109452.html
摘要:比如下面一個(gè)例子例輸出為先輸出,沒有問題,因?yàn)槭峭饺蝿?wù)在主線程中優(yōu)先執(zhí)行,這里的問題是和任務(wù)的執(zhí)行優(yōu)先級(jí)是如何定義的。 在原文的基礎(chǔ)上加了一點(diǎn)參考資料 問題的引出 event loop都不陌生,是指主線程從任務(wù)隊(duì)列中循環(huán)讀取任務(wù),比如 例1: setTimeout(function(){console.log(1)},0); console.log(2) //輸出2,1 在上述...
摘要:回調(diào)函數(shù)這是異步編程最基本的方法。對(duì)象對(duì)象是工作組提出的一種規(guī)范,目的是為異步編程提供統(tǒng)一接口。誕生后,出現(xiàn)了函數(shù),它將異步編程帶入了一個(gè)全新的階段。 更多詳情點(diǎn)擊http://blog.zhangbing.club/Ja... Javascript 語言的執(zhí)行環(huán)境是單線程的,如果沒有異步編程,根本沒法用,非卡死不可。 為了解決這個(gè)問題,Javascript語言將任務(wù)的執(zhí)行模式分成兩種...
摘要:一般會(huì)這樣去寫要在第一個(gè)請(qǐng)求成功后才可以執(zhí)行下一步這樣的寫法的原理是,當(dāng)執(zhí)行一些異步操作時(shí),我們需要知道操作是否已經(jīng)完成,所有當(dāng)執(zhí)行完成的時(shí)候會(huì)返回一個(gè)回調(diào)函數(shù),表示操作已經(jīng)完成。 前言 開篇首先設(shè)想一個(gè)日常開發(fā)常常會(huì)遇到的需求:在多個(gè)接口異步請(qǐng)求數(shù)據(jù),然后利用這些數(shù)據(jù)來進(jìn)行一系列的操作。一般會(huì)這樣去寫: $.ajax({ url: ......, success: f...
摘要:二瀏覽器端在講解事件循環(huán)之前先談?wù)勚型酱a異步代碼的執(zhí)行流程。三端我自己認(rèn)為的事件循環(huán)和瀏覽器端還是有點(diǎn)區(qū)別的,它的事件循環(huán)依靠引擎。四總結(jié)本篇主要介紹了瀏覽器和對(duì)于事件循環(huán)機(jī)制實(shí)現(xiàn),由于能力水平有限,其中可能有誤之處歡迎指出。 一、前言 前幾天聽公司一個(gè)公司三年的前端說今天又學(xué)到了一個(gè)知識(shí)點(diǎn)-微任務(wù)、宏任務(wù),我問他這是什么東西,由于在吃飯他淺淺的說了下,當(dāng)時(shí)沒太理解就私下學(xué)習(xí)整理一...
摘要:異步問題回調(diào)地獄首先,我們來看下異步編程中最常見的一種問題,便是回調(diào)地獄。同時(shí)使用也是異步編程最基礎(chǔ)和核心的一種解決思路。基于,目前也被廣泛運(yùn)用,其是異步編程的一種解決方案,比傳統(tǒng)的回調(diào)函數(shù)解決方案更合理和強(qiáng)大。 關(guān)于 微信公眾號(hào):前端呼啦圈(Love-FED) 我的博客:勞卜的博客 知乎專欄:前端呼啦圈 前言 在實(shí)際編碼中,我們經(jīng)常會(huì)遇到Javascript代碼異步執(zhí)行的場(chǎng)景...
閱讀 1981·2023-04-25 15:45
閱讀 1218·2021-09-29 09:34
閱讀 2507·2021-09-03 10:30
閱讀 2015·2019-08-30 15:56
閱讀 1470·2019-08-29 15:31
閱讀 1274·2019-08-29 15:29
閱讀 3206·2019-08-29 11:24
閱讀 3065·2019-08-26 13:45