JS事件循環(huán)機制event loop宏任務微任務原理解析
首先看一段代碼
async function (){ await f2() console.log('f1') } async function f2(){ console.log('f2') } console.log('正常1') f1() setTimeout(()=>{ console.log('定時器') }) console.log('正常2')
正確的打印順序應該是:正常1,f2 ,正常2,f1,定時器
為什么會出現(xiàn)這樣打印順序呢
首先javascript是一門單線程語言,在最新的HTML5中提出了Web-Worker,但javascript是單線程這一核心仍未改變。既然js是單線程,那就像只有一個窗口的銀行,客戶需要排隊一個一個辦理業(yè)務,同理js任務也要一個一個順序執(zhí)行。如果一個任務耗時過長,那么后一個任務也必須等著。所以就出現(xiàn)了同步任務和異步任務。
概念
除了廣義的同步任務和異步任務,對任務可以進行更精細的區(qū)分
- macro-task(宏任務):包括整體代碼script,setTimeout,setInterval
- micro-task(微任務):Promise,process.nextTick
宏任務:瀏覽器為了能夠使得JS內(nèi)部task與DOM任務能夠有序的執(zhí)行,會在一個task執(zhí)行結(jié)束后,在下一個 task 執(zhí)行開始前,對頁面進行重新渲染 (task->渲染->task->…)
鼠標點擊會觸發(fā)一個事件回調(diào),需要執(zhí)行一個宏任務,然后解析HTMl
微任務:微任務通常來說就是需要在當前 同步任務 執(zhí)行結(jié)束后立即執(zhí)行的任務,比如對一系列動作做出反饋,或者是需要異步的執(zhí)行任務而又不需要分配一個新的任務,這樣便可以減小一點性能的開銷。
既然我們清楚了概念,我們再看一遍代碼
async function (){ await f2() console.log('f1') } async function f2(){ console.log('f2') } console.log('正常1') f1() setTimeout(()=>{ console.log('定時器') }) console.log('正常2')
執(zhí)行順序
首先我們進行正常的同步流程,打印出‘正常1',接下來執(zhí)行f1()函數(shù),await后面的函數(shù)f2()會立即執(zhí)行,所以會打印'f2',繼續(xù)執(zhí)行同步代碼打印‘正常2',至此同步任務全部結(jié)束,開始執(zhí)行異步任務微任務,await f2()等待f2()方法執(zhí)行完之后打印出f1,最后執(zhí)行宏任務setTimeout打印‘定時器'
這就是為什么‘正常1',正常2'會打印在‘f1'之前,因為所有微任務執(zhí)行的時候,當前執(zhí)行棧的代碼必須已經(jīng)執(zhí)行完畢。‘f2','f1'會打印在‘定時器'之前是因為所有微任務總會在下一個宏任務之前全部執(zhí)行完畢
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
JS控制阿拉伯數(shù)字轉(zhuǎn)為中文大寫示例代碼
阿拉伯數(shù)字如何轉(zhuǎn)為中文大寫這個實現(xiàn)的方法有很多,在本文將為大家介紹下js中時如何實現(xiàn)的,感興趣的朋友可以參考下2013-09-09URL的參數(shù)中有加號傳值變?yōu)榭崭竦膯栴}(URL特殊字符)
今天在調(diào)試客戶端向服務器傳遞參數(shù)時,參數(shù)中的“+”全部變成了空格,原因是URL中默認的將“+”號轉(zhuǎn)義了,經(jīng)過以下步驟解決了,需要的朋友可以參考一下2016-11-11使用jsonp實現(xiàn)跨域獲取數(shù)據(jù)實例講解
這篇文章主要介紹了使用jsonp實現(xiàn)跨域獲取數(shù)據(jù)實例講解,需要的朋友可以參考下2016-12-12JS實現(xiàn)仿新浪微博發(fā)布內(nèi)容為空時提示功能代碼
這篇文章主要介紹了JS實現(xiàn)仿新浪微博發(fā)布內(nèi)容為空時提示功能,2015-08-08