js在瀏覽器中的event loop事件隊列示例詳解
前言
以下內(nèi)容是js在瀏覽器中的事件隊列執(zhí)行,與在nodejs中有所區(qū)別,請注意。
都說js是單線程的,不過它本身其實不是單線程,但是在瀏覽器中執(zhí)行時只分配一個線程進行執(zhí)行。
所以說js執(zhí)行是單線程的,一次只能進行一項任務(wù),就是一件任務(wù)一件任務(wù)做,做完一件做下一件。
認識一個棧兩個隊列
一個調(diào)用棧Stack。
一個宏隊列,macrotask,也叫tasks。
一個微隊列,microtask,也叫jobs。
執(zhí)行過程
js就是執(zhí)行全局Script同步代碼,這中間碰到一些異步任務(wù)先加進對應(yīng)的隊列。
做完之后,調(diào)用棧就為空了。
然后將隊列(先微隊列后宏隊列)里面的首個任務(wù)提到調(diào)用棧來做,一件一件做完直到隊列中的任務(wù)都做完。
總結(jié)就是,先做同步的任務(wù),再做微隊列的任務(wù),再做宏隊列的任務(wù)。
異步任務(wù)怎么分配
這些異步任務(wù)包括但不限于:
以下分配給宏隊列:
setTimeout
setInterval
requestAnimationFrame
I/O
UI rendering
以下分配給微隊列:
Promise
Object.observe
MutationObserver
常見的宏隊列:setTimeout,常見的微隊列:Promise。
簡單例子
console.log("同步任務(wù)1"); setTimeout(() => { console.log("宏任務(wù)"); }); new Promise((resolve, reject) => { console.log("同步任務(wù)2"); resolve("微任務(wù)"); }).then((data) => { console.log(data); }); console.log("同步任務(wù)3");
結(jié)果是(按標號加任務(wù),按箭頭執(zhí)行):
需要注意的是Promise的第一層沒有執(zhí)行回調(diào)之前是同步的,也就是上面的同步任務(wù)2。
難一點的例子
console.log("同步任務(wù)1"); console.log("同步任務(wù)2"); new Promise((resolve, reject) => { console.log("同步任務(wù)3"); setTimeout(() => { console.log("宏任務(wù)1"); Promise.resolve() .then(() => { console.log("微任務(wù)5"); }) .then(() => { console.log("微任務(wù)6"); }); }); resolve("微任務(wù)1"); }) .then((data) => { console.log(data); return "微任務(wù)3"; }) .then((data) => { console.log(data); }); setTimeout(() => { console.log("宏任務(wù)2"); }, 0); new Promise((resolve, reject) => { resolve("微任務(wù)2"); }) .then((data, resolve) => { console.log(data); return "微任務(wù)4"; }) .then((data) => { console.log(data); }); console.log("同步任務(wù)4");
如何看呢,先看第一層,紅色代表同步,綠色微任務(wù),藍色宏任務(wù)。
我們會把同步任務(wù)執(zhí)行完,然后看見微任務(wù)有倆,宏任務(wù)也有倆。
本來的執(zhí)行順序可能是這樣(我這里按照序號來表達順序了,請和簡單例子區(qū)分開來):
但是沒那么順利,執(zhí)行到標號6時不一樣了。
因為微任務(wù)執(zhí)行過程中可能會產(chǎn)生新的微任務(wù)。
上面的微任務(wù)1執(zhí)行完會把微任務(wù)3加在微任務(wù)2后面,也就是微任務(wù)2執(zhí)行完也輪不到宏任務(wù),會繼續(xù)執(zhí)行新的微任務(wù)直到微任務(wù)隊列暫時為空。
所以接下來會按照加入隊列的順序執(zhí)行完四個微任務(wù),這時候發(fā)現(xiàn)沒有新的微任務(wù)產(chǎn)生,才開始執(zhí)行宏任務(wù):
但是需要注意的是,上面執(zhí)行到標號5
時又不一樣了,宏任務(wù)一執(zhí)行后又產(chǎn)生了新的微任務(wù),所以宏任務(wù)兩個并沒有順利連續(xù)執(zhí)行,而是被插入的微任務(wù)攔住了。
(要記住微任務(wù)與宏任務(wù)隊列都存在時一定是微任務(wù)先執(zhí)行完再來執(zhí)行宏任務(wù),即使是宏任務(wù)執(zhí)行產(chǎn)生的微任務(wù)也同理)
所以最后的答案,如果存在不理解的,可以在認真回顧一下上文:
總結(jié)
到此這篇關(guān)于js在瀏覽器中的event loop事件隊列的文章就介紹到這了,更多相關(guān)js event loop事件隊列內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
詳述 Sublime Text 打開 GBK 格式中文亂碼的解決方法
這篇文章主要介紹了詳述 Sublime Text 打開 GBK 格式中文亂碼的解決方法,非常具有實用價值,需要的朋友可以參考下2017-10-10JS實現(xiàn)的圖片選擇順序切換和循環(huán)切換功能示例【測試可用】
這篇文章主要介紹了JS實現(xiàn)的圖片選擇順序切換和循環(huán)切換功能,結(jié)合完整實例形式分析了JavaScript基于事件響應(yīng)與樣式動態(tài)操作實現(xiàn)圖片切換相關(guān)操作技巧,需要的朋友可以參考下2018-12-12javascript跨域總結(jié)之window.name實現(xiàn)的跨域數(shù)據(jù)傳輸
本文給大家介紹window.name實現(xiàn)的跨域數(shù)據(jù)傳輸,自己親自實踐了一下,真的非常好用,特此分享到腳本之家網(wǎng)站供大家參考2015-11-11微信小程序如何調(diào)用新聞接口實現(xiàn)列表循環(huán)
這篇文章主要介紹了微信小程序如何調(diào)用新聞接口實現(xiàn)列表循環(huán),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習或者工作具有一定的參考學(xué)習價值,需要的朋友可以參考下2019-07-07