摘要:瀏覽器與的異同,以及部分機(jī)制有人對部分迷惑,本身構(gòu)造函數(shù)是同步的,是異步。瀏覽器的的已全部分析完成,過程中引用阮一峰博客,知乎,部分文章內(nèi)容,侵刪。
瀏覽器與NodeJS的EventLoop異同,以及部分機(jī)制
PS:有人對promise部分迷惑,Promise本身構(gòu)造函數(shù)是同步的,.then是異步。---- 2018/7/6 22:35修改
javascript 是一門單線程的腳本語言,雖然是單線程但是有很多異步的API來幫助開發(fā)者解決線程的阻塞問題。比如:onClick 注冊的回調(diào)函數(shù)、必不可少的ajax等等...但是 javascript 運(yùn)行環(huán)境是如何做到單線程卻又不是一直阻塞線程等待各種異步操作完成才繼續(xù)執(zhí)行操作的呢?
答案就是: event loop
1.event loop 的規(guī)范是在HTML5中規(guī)定的。 2.event loop 是 javascript 運(yùn)行環(huán)境(手動加粗) 的機(jī)制。 3.瀏覽器實(shí)現(xiàn)的event loop 與 NodeJS 實(shí)現(xiàn)的event loop 是有異同的。
HTML5 中定義 event loop 規(guī)范鏈接 https://www.w3.org/TR/html5/w...
一 瀏覽器的event loop
1.簡單了解
event loop 即事件循環(huán),它到底是什么結(jié)構(gòu)呢? 阮一峰老師的博客有一張圖,雖然很直白、明了但是少了一些東西不能全面的將 event loop 整體循環(huán)機(jī)制展示出來。先來看圖:
圖片非筆者原創(chuàng),來自阮一峰博客,在此說明,侵刪。
從圖中我們可以得到信息是:
1.javascript 引擎執(zhí)行 javascript 是單線程的,因?yàn)橹挥幸粋€ stack 里面有各種正在執(zhí)行、等待執(zhí)行的事件。
2.有一些 webAPI 將執(zhí)行時產(chǎn)生的 callback 放入一個隊(duì)列,即 “事件隊(duì)列”。
3.在event loop 循環(huán)中不停的將“事件隊(duì)列”里等待執(zhí)行的事件,推入 javascript 執(zhí)行棧。
這就是事件循環(huán)簡化的機(jī)制,為什么說簡化呢?因?yàn)樵谘h(huán)中還做了很多沒有提及的操作、規(guī)則。
我就不舉栗子了,但是我要打個比方。
就說一個老生常談的問題 (文章編輯不便,直接一行了,換行黨你倒是來打我啊!)
setTimeout(e=>{ console.log(1) },0); new Promise((res,rej)=>{ res() }).then(e=>{ console.log(2) });
同樣都是 javascript 中提供的異步API,同樣都是直接執(zhí)行( 開發(fā)者所希望的,雖然會因?yàn)樽枞麑?dǎo)致延時,防止杠精 ),但是不論這倆行代碼誰上、誰下,輸出都會是 2 1。因?yàn)檫@里涉及 event loop 中 macro task 與 micro task 的執(zhí)行順序、規(guī)則。
2.整體流程
回到剛才說那張流程圖不夠完善的問題上,現(xiàn)在來一張完整的、全面的 event loop 流程圖。
圖片非筆者原創(chuàng),來secrets of javascript ninja,在此說明,侵刪。
這是一個 event loop 完整的流程圖,從圖中我們看到了許多剛才未提及的名詞,從頭到尾的梳理一遍 (從上至下):
1.讀取 Macrotask queue 中任務(wù)。有倆種情況
任務(wù)隊(duì)列空,向下執(zhí)行
任務(wù)隊(duì)列不為空,將最先進(jìn)入的一個(手動+文章加粗)任務(wù)推入 javascript 執(zhí)行棧,向下執(zhí)行
2.讀取 Microtask queue 中任務(wù)。有倆種情況
任務(wù)隊(duì)列空,向下執(zhí)行
任務(wù)隊(duì)列不為空,將最先進(jìn)入的一個任務(wù)推入 javascript 執(zhí)行棧,并且再次重復(fù)此操作(手動+文章加粗),直到 Microtask queue 為空。直白的說:將此任務(wù)隊(duì)列按照先后順序?qū)⑺腥蝿?wù)推入javascript 執(zhí)行棧,向下執(zhí)行
3.根據(jù)本次循環(huán)耗時(手動+文章加粗)判斷是否需要、是否可以更新UI 【 后面會提一下這個循環(huán)時間問題 】
不需要,重復(fù)第一步
需要,向下執(zhí)行
4.更新UI,UI rendering,同時阻塞 javascript 執(zhí)行。并且繼續(xù)重復(fù)第一步。
以上便是一整個 event loop 流程,從流程中我們可以看到有倆個“任務(wù)隊(duì)列”,這倆個隊(duì)列實(shí)例化到 javascript 中的API 便是
Macrotask queue --> setTimeout || setInterval || javascript代碼 Microtask queue --> Promise.then()
至此一個完整的 event loop 流程便完全說完了。
3.實(shí)例解析
什么鬼?這么復(fù)雜? 弄懂?不存在的
現(xiàn)在回到剛才提到的 “老生常談的問題” 從實(shí)例的角度來說明一下問題。我們假設(shè)這個 javascript 文件叫做 "main.js"
"main.js"中的代碼(+ 為自定義標(biāo)記)
+1 console.log(1); +2 setTimeout(e=>{ console.log(2); },0) +3 setTimeout(e=>{ console.log(3); },0) +4 new Promise((resolve,reject)=>{ console.log(4); resolve();}) .then(e=>{ console.log(5); }) +5 setTimeout(e=>{ console.log(6); +6 new Promise((resolve,reject)=>{ console.log(7); resolve(); }) .then(e=>{ console.log(8);}) })
那么這個執(zhí)行順序是怎樣呢?從頭帶尾梳理一遍(詞窮,全文只要是流程統(tǒng)一是“從頭到尾梳理一遍”)
macrotask: javascript 代碼,所有同步代碼執(zhí)行。輸出:1 4。注冊 +4 到 microtask。 注冊+2 +3 +5 到 macrotask。
microtask: 執(zhí)行 +4 輸出:5。
macrotask: 執(zhí)行 +2。 輸出 2。
microtask: 無
macrotask: 執(zhí)行 +3。 輸出 3。
microtask: 無
macrotask: 執(zhí)行 +5。 輸出 6 7。 注冊 +6 到 microtask。
microtask: 輸出 8。
所以總體輸出的順序?yàn)椋?strong>1 4 5 2 3 6 7 8
如果這個輸出與你所想相同,那么基本就沒有問題了。
那么如果不對或者有問題怎么辦?
PS: 前面提到 【本次循環(huán)耗時】這個問題,這里我也不是非常清楚,望大牛指點(diǎn)。瀏覽器一般渲染頁面60/S,以達(dá)到每秒60幀(60 fps),所以大概16ms一次,既然有了時間我們不經(jīng)就會問?前面的任務(wù)處理耽誤了則么辦?因?yàn)閖avascript線程與UI線程互斥,某些任務(wù)導(dǎo)致 javascript引擎 坑了隊(duì)友,自然而然沒法在16ms的節(jié)點(diǎn)上到達(dá)這一步,從secrets of javascript ninja中了解到,一般會摒棄這次渲染,等待下一次循環(huán)。( 如有問題請指正! )
瀏覽器中的 event loop 到此結(jié)束,下面說說 NodeJS 的 event loop
二 NodeJS的event loop
NodeJS 的 event loop 也是有 Macrotask queue 與 Microtask queue 的。只不過 NodeJS 的略有不同。那么主要說說不同在哪里。
NodeJS中 Macrotask queue 與 Microtask queue 實(shí)例化到API為: Macrotask queue --> script(主程序代碼),setImmediate, I/O,setTimeout, setInterval Microtask queue --> process.nextTick, Promise
1.Macrotask queue 不同之處
上面說到了瀏覽器 event loop 的 Macrotask queue 在每次循環(huán)中只會讀取一個任務(wù),NodeJS 中 Macrotask queue 會一次性讀取完畢( 同階段的執(zhí)行完畢,后面會說到Macrotask queue 分為 6個階段 ),然后向下讀取Microtask。
注意: 這一條與 NodeJS版本有很大關(guān)系,在看 深入淺出NodeJS 這一本書時( 看的版本很舊,不知是否有修訂版,如有請告知。 ),提到的 setImmediate 每次循環(huán)只會執(zhí)行一次,并且給出的示例在 v8.9.1 版本跑時已不符合書中所寫。書中示例如下(+ 為自定義標(biāo)記,原文中沒有):
+1 process.nextTick(function () { console.log("nextTick執(zhí)行1"); }); +2 process.nextTick(function () { console.log("nextTick執(zhí)行2"); }); +3 setImmediate(function () { console.log("setImmediate?執(zhí)行1"); +4 process.nextTick(function () { console.log("強(qiáng)勢插入"); }); }); +5 setImmediate(function () { console.log("setImmediate?執(zhí)行2"); }); +6 console.log("正常執(zhí)行"); 正常執(zhí)行 nextTick執(zhí)行1 nextTick執(zhí)行2 setImmediate執(zhí)行1 強(qiáng)勢插入 setImmediate?執(zhí)行2
在 v8.9.1 中截圖如下
從圖片中可以看到,至少在 v8.9.1 版本中 Macrotask queue 會直接全部執(zhí)行。按照慣例從頭到尾的梳理一遍:
macrotask: javascript 代碼,所有同步代碼執(zhí)行。輸出:正常執(zhí)行。注冊 +3 +5 到 Macrotask。執(zhí)行process.nextTick(),最終輸出:正常執(zhí)行, nextTick執(zhí)行1, nextTick執(zhí)行2。
**microtask: 無
macrotask: 執(zhí)行 +3 +5。 輸出:setImmediate執(zhí)行1, setImmediate?執(zhí)行2。 執(zhí)行process.nextTick(),最終輸出:setImmediate執(zhí)行1, setImmediate?執(zhí)行2,強(qiáng)勢插入。
microtask: 無
所以最終輸出為:正常執(zhí)行, nextTick執(zhí)行1, nextTick執(zhí)行2,setImmediate執(zhí)行1, setImmediate?執(zhí)行2,強(qiáng)勢插入。
2.process.nextTick(),setImmediates,以及event loop的6個階段
NodeJS 中 Macrotask queue會分為 6 個階段,每個階段的作用如下(process.nextTick()在6個階段結(jié)束的時候都會執(zhí)行):
timers:執(zhí)行setTimeout() 和 setInterval()中到期的callback。 I/O callbacks:上一輪循環(huán)中有少數(shù)的I/Ocallback會被延遲到這一輪的這一階段執(zhí)行 idle, prepare:僅內(nèi)部使用 poll:最為重要的階段,執(zhí)行I/O callback,在適當(dāng)?shù)臈l件下會阻塞在這個階段 check:執(zhí)行setImmediate的callback close callbacks:執(zhí)行close事件的callback,例如socket.on("close",func)
注:此6個階段非筆者原創(chuàng)來自 https://cnodejs.org/topic/5a9...,文章從底層C代碼分析NodeJS event loop。這里做只做簡單整合。侵刪。
在了解了這六個階段后,我們可以發(fā)現(xiàn)定時器系列在NodeJS event loop中 Macrotask queue 讀取順序?yàn)椋?/p>
1. setTimeout(fun,0) setInterval(fun,0) 2. setImmediate
空口無憑,在實(shí)例中了解。的代碼奉上( 代碼較長,分為三段,方便閱讀,避免滾動。 ):
+1 process.nextTick(function(){ console.log("1"); }); +2 process.nextTick(function(){ console.log("2"); +3 setImmediate(function(){ console.log("3"); }); +4 process.nextTick(function(){ console.log("4"); }); }); +5 setImmediate(function(){ console.log("5"); +6 process.nextTick(function(){ console.log("6"); }); +7 setImmediate(function(){ console.log("7"); }); });
+8 setTimeout(e=>{ console.log(8); +9 new Promise((resolve,reject)=>{ console.log(8+"promise"); resolve(); }).then(e=>{ console.log(8+"promise+then"); }) },0) +10 setTimeout(e=>{ console.log(9); },0) +11 setImmediate(function(){ console.log("10"); +12 process.nextTick(function(){ console.log("11"); }); +13 process.nextTick(function(){ console.log("12"); }); +14 setImmediate(function(){ console.log("13"); }); });
console.log("14"); +15 new Promise((resolve,reject)=>{ console.log(15); resolve(); }).then(e=>{ console.log(16); })
這么復(fù)雜的異步嵌套在一起是不是很頭疼呢?
我!不!看!了!
最后一遍梳理,最多、最全的一次梳理。自古以來從頭到尾的梳理一遍
macrotask: javascript 代碼,所有同步代碼執(zhí)行。輸出:14。執(zhí)行process.nextTick(),最終輸出:14,15, 1, 2, 4。 注冊 +3 +5 +8 +11 到 Macrotask。 注冊 +15 到 Microtask。
microtask: 執(zhí)行 +15 輸出 16
macrotask: 執(zhí)行 +8 +10 輸出 8, 8promise, 9。 注冊 +9 到 Microtask。
microtask: 執(zhí)行 +9 輸出 8promise+then
macrotask: 執(zhí)行 +5 +11 +3 輸出 5, 10, 3。 注冊 +7 +14 到 macrotask。執(zhí)行process.nextTick(),最終輸出:5 10 3 6 11 12。
microtask: 無
macrotask: 執(zhí)行 +7 +14。 輸出:7,13
microtask: 無
由此最中全部的輸出為:14,15,1,2,4,8,8promise,9,8promise+then,5,10,3,6,11,12,7,13
三 結(jié)束
到此結(jié)束了。瀏覽器的、NodeJS 的 event loop 已全部分析完成,過程中引用:阮一峰博客,知乎,CSDN部分文章內(nèi)容,侵刪。
最近在了解部分底層知識,收獲頗豐。其中包括 for of.... 等等各種奇奇怪怪的問題,有時間再寫吧。
最后,本人菜鳥,如有不對、不實(shí)、誤導(dǎo)等錯誤、問題,歡迎評論區(qū)指正。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/95971.html
摘要:在他的重學(xué)前端課程中提到到現(xiàn)在為止,前端工程師已經(jīng)成為研發(fā)體系中的重要崗位之一。大部分前端工程師的知識,其實(shí)都是來自于實(shí)踐和工作中零散的學(xué)習(xí)。一基礎(chǔ)前端工程師吃飯的家伙,深度廣度一樣都不能差。 開篇 前端開發(fā)是一個非常特殊的行業(yè),它的歷史實(shí)際上不是很長,但是知識之繁雜,技術(shù)迭代速度之快是其他技術(shù)所不能比擬的。 winter在他的《重學(xué)前端》課程中提到: 到現(xiàn)在為止,前端工程師已經(jīng)成為研...
摘要:在他的重學(xué)前端課程中提到到現(xiàn)在為止,前端工程師已經(jīng)成為研發(fā)體系中的重要崗位之一。大部分前端工程師的知識,其實(shí)都是來自于實(shí)踐和工作中零散的學(xué)習(xí)。一基礎(chǔ)前端工程師吃飯的家伙,深度廣度一樣都不能差。開篇 前端開發(fā)是一個非常特殊的行業(yè),它的歷史實(shí)際上不是很長,但是知識之繁雜,技術(shù)迭代速度之快是其他技術(shù)所不能比擬的。 winter在他的《重學(xué)前端》課程中提到: 到現(xiàn)在為止,前端工程師已經(jīng)成為研發(fā)體系...
摘要:下面我們說一下前端的和?,F(xiàn)在就可以知道了,前端的其實(shí)是由組合而成。這么一對比,相信很多小伙伴對更加了解了,原來前端和服務(wù)端的如此相似,他們的基礎(chǔ)是相同的,只是環(huán)境不同,導(dǎo)致他們擴(kuò)展出來的東西不同而已。 前言 很多小伙伴學(xué)Node的時候,都沒有好好認(rèn)識她就開始瘋狂追求,想一舉拿下,直接在網(wǎng)上搜索Node實(shí)戰(zhàn),想知道她活好不好,想先用她建個簡單博客練練手。 JavaScript和Nodej...
閱讀 3563·2021-11-22 15:11
閱讀 4643·2021-11-18 13:15
閱讀 2710·2019-08-29 14:08
閱讀 3583·2019-08-26 13:49
閱讀 3100·2019-08-26 12:17
閱讀 3294·2019-08-26 11:54
閱讀 3118·2019-08-26 10:58
閱讀 2038·2019-08-26 10:21