摘要:就是每次傳入的函數(shù)最后是的任務(wù)之后,開始執(zhí)行,可以看到此時會批量執(zhí)行中的函數(shù),而且還給這些中回調(diào)函數(shù)放入了一個這個很顯眼的函數(shù)之中,表示這些回調(diào)函數(shù)是在微任務(wù)中執(zhí)行的。下一模塊會對此微任務(wù)中的插隊行為進行詳解。
有關(guān)Eventloop+Promise的面試題大約分以下幾個版本——得心應(yīng)手版、游刃有余版、爐火純青版、登峰造極版和究極{{BANNED}}版。假設(shè)小伙伴們戰(zhàn)到最后一題,以后遇到此類問題,都是所向披靡。當(dāng)然如果面試官們還能想出更{{BANNED}}的版本,算我輸。
版本一:得心應(yīng)手版考點:eventloop中的執(zhí)行順序,宏任務(wù)微任務(wù)的區(qū)別。吐槽:這個不懂,沒得救了,回家重新學(xué)習(xí)吧。
setTimeout(()=>{ console.log(1) },0) Promise.resolve().then(()=>{ console.log(2) }) console.log(3)
這個版本的面試官們就特別友善,僅僅考你一個概念理解,了解宏任務(wù)(marcotask)微任務(wù)(microtask),這題就是送分題。
筆者答案:這個是屬于Eventloop的問題。main script運行結(jié)束后,會有微任務(wù)隊列和宏任務(wù)隊列。微任務(wù)先執(zhí)行,之后是宏任務(wù)。
PS:概念問題
有時候會有版本是宏任務(wù)>微任務(wù)>宏任務(wù),在這里筆者需要講清楚一個概念,以免混淆。這里有個main script的概念,就是一開始執(zhí)行的代碼(代碼總要有開始執(zhí)行的時候?qū)Π桑蝗缓耆蝿?wù)和微任務(wù)的隊列哪里來的),這里被定義為了宏任務(wù)(筆者喜歡將main script的概念多帶帶拎出來,不和兩個任務(wù)隊列混在一起),然后根據(jù)main script中產(chǎn)生的微任務(wù)隊列和宏任務(wù)隊列,分別清空,這個時候是先清空微任務(wù)的隊列,再去清空宏任務(wù)的隊列。
版本二:游刃有余版這一個版本,面試官們?yōu)榱丝简炓幌聦τ赑romise的理解,會給題目加點料:
考點:Promise的executor以及then的執(zhí)行方式吐槽:這是個小坑,promise掌握的熟練的,這就是人生的小插曲。
setTimeout(()=>{ console.log(1) },0) let a=new Promise((resolve)=>{ console.log(2) resolve() }).then(()=>{ console.log(3) }).then(()=>{ console.log(4) }) console.log(4)
此題看似在考Eventloop,實則考的是對于Promise的掌握程度。Promise的then是微任務(wù)大家都懂,但是這個then的執(zhí)行方式是如何的呢,以及Promise的executor是異步的還是同步的?
錯誤示范:Promise的then是一個異步的過程,每個then執(zhí)行完畢之后,就是一個新的循環(huán)的,所以第二個then會在setTimeout之后執(zhí)行。(沒錯,這就是某年某月某日筆者的一個回答。請給我一把槍,真想打死當(dāng)時的自己。)
正確示范:這個要從Promise的實現(xiàn)來說,Promise的executor是一個同步函數(shù),即非異步,立即執(zhí)行的一個函數(shù),因此他應(yīng)該是和當(dāng)前的任務(wù)一起執(zhí)行的。而Promise的鏈?zhǔn)秸{(diào)用then,每次都會在內(nèi)部生成一個新的Promise,然后執(zhí)行then,在執(zhí)行的過程中不斷向微任務(wù)(microtask)推入新的函數(shù),因此直至微任務(wù)(microtask)的隊列清空后才會執(zhí)行下一波的macrotask。詳細解析
(如果大家不嫌棄,可以參考我的另一篇文章,從零實現(xiàn)一個Promise,里面的解釋淺顯易懂。)
我們以babel的core-js中的promise實現(xiàn)為例,看一眼promise的執(zhí)行規(guī)范:
代碼位置:promise-polyfill
PromiseConstructor = function Promise(executor) { //... try { executor(bind(internalResolve, this, state), bind(internalReject, this, state)); } catch (err) { internalReject(this, state, err); } };
這里可以很清除地看到Promise中的executor是一個立即執(zhí)行的函數(shù)。
then: function then(onFulfilled, onRejected) { var state = getInternalPromiseState(this); var reaction = newPromiseCapability(speciesConstructor(this, PromiseConstructor)); reaction.ok = typeof onFulfilled == "function" ? onFulfilled : true; reaction.fail = typeof onRejected == "function" && onRejected; reaction.domain = IS_NODE ? process.domain : undefined; state.parent = true; state.reactions.push(reaction); if (state.state != PENDING) notify(this, state, false); return reaction.promise; },
接著是Promise的then函數(shù),很清晰地看到reaction.promise,也就是每次then執(zhí)行完畢后會返回一個新的Promise。也就是當(dāng)前的微任務(wù)(microtask)隊列清空了,但是之后又開始添加了,直至微任務(wù)(microtask)隊列清空才會執(zhí)行下一波宏任務(wù)(marcotask)。
//state.reactions就是每次then傳入的函數(shù) var chain = state.reactions; microtask(function () { var value = state.value; var ok = state.state == FULFILLED; var i = 0; var run = function (reaction) { //... }; while (chain.length > i) run(chain[i++]); //... });
最后是Promise的任務(wù)resolve之后,開始執(zhí)行then,可以看到此時會批量執(zhí)行then中的函數(shù),而且還給這些then中回調(diào)函數(shù)放入了一個microtask這個很顯眼的函數(shù)之中,表示這些回調(diào)函數(shù)是在微任務(wù)中執(zhí)行的。
那么在沒有Promise的瀏覽器中,微任務(wù)這個隊列是如何實現(xiàn)的呢?
小知識:babel中對于微任務(wù)的polyfill,如果是擁有setImmediate函數(shù)平臺,則使用之,若沒有則自定義則利用各種比如nodejs中的process.nextTick,瀏覽器中支持postMessage的,或者是通過create一個script來實現(xiàn)微任務(wù)(microtask)。最終的最終,是使用setTimeout,不過這個就和微任務(wù)無關(guān)了,promise變成了宏任務(wù)的一員。
拓展思考:
為什么有時候,then中的函數(shù)是一個數(shù)組?有時候就是一個函數(shù)?
我們稍稍修改一下上述題目,將鏈?zhǔn)秸{(diào)用的函數(shù),變成下方的,分別調(diào)用then。且不說這和鏈?zhǔn)秸{(diào)用之間的不同用法,這邊只從實踐角度辨別兩者的不同。鏈?zhǔn)秸{(diào)用是每次都生成一個新的Promise,也就是說每個then中回調(diào)方法屬于一個microtask,而這種分別調(diào)用,會將then中的回調(diào)函數(shù)push到一個數(shù)組之中,然后批量執(zhí)行。再換句話說,鏈?zhǔn)秸{(diào)用可能會被Evenloop中其他的函數(shù)插隊,而分別調(diào)用則不會(僅針對最普通的情況,then中無其他異步操作。)。
let a=new Promise((resolve)=>{ console.log(2) resolve() }) a.then(()=>{ console.log(3) }) a.then(()=>{ console.log(4) })
下一模塊會對此微任務(wù)(microtask)中的“插隊”行為進行詳解。
版本三:爐火純青版這一個版本是上一個版本的進化版本,上一個版本的promise的then函數(shù)并未返回一個promise,如果在promise的then中創(chuàng)建一個promise,那么結(jié)果該如何呢?
考點:promise的進階用法,對于then中return一個promise的掌握吐槽:promise也可以是地獄……
new Promise((resolve,reject)=>{ console.log("promise1") resolve() }).then(()=>{ console.log("then11") new Promise((resolve,reject)=>{ console.log("promise2") resolve() }).then(()=>{ console.log("then21") }).then(()=>{ console.log("then23") }) }).then(()=>{ console.log("then12") })
按照上一節(jié)最后一個microtask的實現(xiàn)過程,也就是說一個Promise所有的then的回調(diào)函數(shù)是在一個microtask函數(shù)中執(zhí)行的,但是每一個回調(diào)函數(shù)的執(zhí)行,又按照情況分為立即執(zhí)行,微任務(wù)(microtask)和宏任務(wù)(macrotask)。
遇到這種嵌套式的Promise不要慌,首先要心中有一個隊列,能夠?qū)⑦@些函數(shù)放到相對應(yīng)的隊列之中。
Ready GO
第一輪
current task: promise1是當(dāng)之無愧的立即執(zhí)行的一個函數(shù),參考上一章節(jié)的executor,立即執(zhí)行輸出[promise1]
micro task queue: [promise1的第一個then]
第二輪
current task: then1執(zhí)行中,立即輸出了then11以及新promise2的promise2
micro task queue: [新promise2的then函數(shù),以及promise1的第二個then函數(shù)]
第三輪
current task: 新promise2的then函數(shù)輸出then21和promise1的第二個then函數(shù)輸出then12。
micro task queue: [新promise2的第二then函數(shù)]
第四輪
current task: 新promise2的第二then函數(shù)輸出then23
micro task queue: []
END
最終結(jié)果[promise1,then11,promise2,then21,then12,then23]。
變異版本1:如果說這邊的Promise中then返回一個Promise呢??
new Promise((resolve,reject)=>{ console.log("promise1") resolve() }).then(()=>{ console.log("then11") return new Promise((resolve,reject)=>{ console.log("promise2") resolve() }).then(()=>{ console.log("then21") }).then(()=>{ console.log("then23") }) }).then(()=>{ console.log("then12") })
這里就是Promise中的then返回一個promise的狀況了,這個考的重點在于Promise而非Eventloop了。這里就很好理解為何then12會在then23之后執(zhí)行,這里Promise的第二個then相當(dāng)于是掛在新Promise的最后一個then的返回值上。
變異版本2:如果說這邊不止一個Promise呢,再加一個new Promise是否會影響結(jié)果??
new Promise((resolve,reject)=>{ console.log("promise1") resolve() }).then(()=>{ console.log("then11") new Promise((resolve,reject)=>{ console.log("promise2") resolve() }).then(()=>{ console.log("then21") }).then(()=>{ console.log("then23") }) }).then(()=>{ console.log("then12") }) new Promise((resolve,reject)=>{ console.log("promise3") resolve() }).then(()=>{ console.log("then31") })
笑容逐漸{{BANNED}},同樣這個我們可以自己心中排一個隊列:
第一輪
current task: promise1,promise3
micro task queue: [promise2的第一個then,promise3的第一個then]
第二輪
current task: then11,promise2,then31
micro task queue: [promise2的第一個then,promise1的第二個then]
第三輪
current task: then21,then12
micro task queue: [promise2的第二個then]
第四輪
current task: then23
micro task queue: []
最終輸出:[promise1,promise3,then11,promise2,then31,then21,then12,then23]
版本四:登峰造極版考點:在async/await之下,對Eventloop的影響。槽點:別被async/await給騙了,這題不難。
相信大家也看到過此類的題目,我這里有個相當(dāng)簡易的解釋,不知大家是否有興趣。
async function async1() { console.log("async1 start"); await async2(); console.log("async1 end"); } async function async2() { console.log( "async2"); } console.log("script start"); setTimeout(function () { console.log("settimeout"); },0); async1(); new Promise(function (resolve) { console.log("promise1"); resolve(); }).then(function () { console.log("promise2"); }); console.log("script end");
async/await僅僅影響的是函數(shù)內(nèi)的執(zhí)行,而不會影響到函數(shù)體外的執(zhí)行順序。也就是說async1()并不會阻塞后續(xù)程序的執(zhí)行,await async2()相當(dāng)于一個Promise,console.log("async1 end");相當(dāng)于前方Promise的then之后執(zhí)行的函數(shù)。
按照上章節(jié)的解法,最終輸出結(jié)果:[script start,async1 start,async2,promise1,script end,async1 end,promise2,settimeout]
如果了解async/await的用法,則并不會覺得這題是困難的,但若是不了解或者一知半解,那么這題就是災(zāi)難啊。
此處唯一有爭議的就是async的then和promise的then的優(yōu)先級的問題,請看下方詳解。*
async/await與promise的優(yōu)先級詳解async function async1() { console.log("async1 start"); await async2(); console.log("async1 end"); } async function async2() { console.log( "async2"); } // 用于test的promise,看看await究竟在何時執(zhí)行 new Promise(function (resolve) { console.log("promise1"); resolve(); }).then(function () { console.log("promise2"); }).then(function () { console.log("promise3"); }).then(function () { console.log("promise4"); }).then(function () { console.log("promise5"); });
先給大家出個題,如果讓你polyfill一下async/await,大家會怎么polyfill上述代碼?下方先給出筆者的版本:
function promise1(){ return new Promise((resolve)=>{ console.log("async1 start"); promise2().then(()=>{ console.log("async1 end"); resolve() }) }) } function promise2(){ return new Promise((resolve)=>{ console.log( "async2"); resolve() }) }
在筆者看來,async本身是一個Promise,然后await肯定也跟著一個Promise,那么新建兩個function,各自返回一個Promise。接著function promise1中需要等待function promise2中Promise完成后才執(zhí)行,那么就then一下咯~。
根據(jù)這個版本得出的結(jié)果:[async1 start,async2,promise1,async1 end,promise2,...],async的await在test的promise.then之前,其實也能夠從筆者的polifill中得出這個結(jié)果。
然后讓筆者驚訝的是用原生的async/await,得出的結(jié)果與上述polyfill不一致!得出的結(jié)果是:[async1 start,async2,promise1,promise2,promise3,async1 end,...],由于promise.then每次都是一輪新的microtask,所以async是在2輪microtask之后,第三輪microtask才得以輸出(關(guān)于then請看版本三的解釋)。
/ 突如其來的沉默 /
這里插播一條,async/await因為要經(jīng)過3輪的microtask才能完成await,被認為開銷很大,因此之后V8和Nodejs12開始對此進行了修復(fù),詳情可以看github上面這一條pull
那么,筆者換一種方式來polyfill,相信大家都已經(jīng)充分了解await后面是一個Promise,但是假設(shè)這個Promise不是好Promise怎么辦?異步是好異步,Promise不是好Promise。V8就很兇殘,加了額外兩個Promise用于解決這個問題,簡化了下源碼,大概是下面這個樣子:
// 不太準(zhǔn)確的一個描述 function promise1(){ console.log("async1 start"); // 暗中存在的promise,筆者認為是為了保證async返回的是一個promise const implicit_promise=Promise.resolve() // 包含了await的promise,這里直接執(zhí)行promise2,為了保證promise2的executor是同步的感覺 const promise=promise2() // https://tc39.github.io/ecma262/#sec-performpromisethen // 25.6.5.4.1 // throwaway,為了規(guī)范而存在的,為了保證執(zhí)行的promise是一個promise const throwaway= Promise.resolve() //console.log(throwaway.then((d)=>{console.log(d)})) return implicit_promise.then(()=>{ throwaway.then(()=>{ promise.then(()=>{ console.log("async1 end"); }) }) }) }
ps:為了強行推遲兩個microtask執(zhí)行,筆者也是煞費苦心。
總結(jié)一下:async/await有時候會推遲兩輪microtask,在第三輪microtask執(zhí)行,主要原因是瀏覽器對于此方法的一個解析,由于為了解析一個await,要額外創(chuàng)建兩個promise,因此消耗很大。后來V8為了降低損耗,所以剔除了一個Promise,并且減少了2輪microtask,所以現(xiàn)在最新版本的應(yīng)該是“零成本”的一個異步。版本五:究極{{BANNED}}版
饕餮大餐,什么{{BANNED}}的內(nèi)容都往里面加,想想就很豐盛。能考到這份上,只能說面試官人狠話也多。
考點:nodejs事件+Promise+async/await+佛系setImmediate槽點:筆者都不知道那個可能先出現(xiàn)
async function async1() { console.log("async1 start"); await async2(); console.log("async1 end"); } async function async2() { console.log( "async2"); } console.log("script start"); setTimeout(function () { console.log("settimeout"); }); async1() new Promise(function (resolve) { console.log("promise1"); resolve(); }).then(function () { console.log("promise2"); }); setImmediate(()=>{ console.log("setImmediate") }) process.nextTick(()=>{ console.log("process") }) console.log("script end");
隊列執(zhí)行start
第一輪:
current task:"script start","async1 start","async2","promise1",“script end”
micro task queue:[async,promise.then,process]
macro task queue:[setTimeout,setImmediate]
第二輪
current task:process,async1 end ,promise.then
micro task queue:[]
macro task queue:[setTimeout,setImmediate]
第三輪
current task:setTimeout,setImmediate
micro task queue:[]
macro task queue:[]
最終結(jié)果:[script start,async1 start,async2,promise1,script end,process,async1 end,promise2,setTimeout,setImmediate]
同樣"async1 end","promise2"之間的優(yōu)先級,因平臺而異。
筆者干貨總結(jié)在處理一段evenloop執(zhí)行順序的時候:
第一步確認宏任務(wù),微任務(wù)
宏任務(wù):script,setTimeout,setImmediate,promise中的executor
微任務(wù):promise.then,process.nextTick
第二步解析“攔路虎”,出現(xiàn)async/await不要慌,他們只在標(biāo)記的函數(shù)中能夠作威作福,出了這個函數(shù)還是跟著大部隊的潮流。
第三步,根據(jù)Promise中then使用方式的不同做出不同的判斷,是鏈?zhǔn)竭€是分別調(diào)用。
最后一步記住一些特別事件
比如,process.nextTick優(yōu)先級高于Promise.then
參考網(wǎng)址,推薦閱讀:有關(guān)V8中如何實現(xiàn)async/await的,更快的異步函數(shù)和 Promise
有關(guān)async/await規(guī)范的,ecma262
還有babel-polyfill的源碼,promise
后記Hello~Anybody here?
本來筆者是不想寫這篇文章的,因為有種5年高考3年模擬的既視感,奈何面試官們都太兇殘了,為了“折磨”面試者無所不用其極,怎么{{BANNED}}怎么來。不過因此筆者算是徹底掌握了Eventloop的用法,因禍得福吧~
有小伙伴看到最后嘛?來和筆者聊聊你遇到過的的Eventloop+Promise的{{BANNED}}題目。
歡迎轉(zhuǎn)載~但請注明出處~首發(fā)于掘金~Eventloop不可怕,可怕的是遇上Promise
題外話:來segmentfault試水~啊哈哈哈啊哈哈
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://specialneedsforspecialkids.com/yun/103054.html
摘要:記錄下我遇到的面試題,都有大佬分享過,附上各個大佬的文章,總結(jié)出其中的主要思想即可。推薦黑金團隊的文章前端緩存最佳實踐推薦名揚的文章淺解強緩存和協(xié)商緩存狀態(tài)碼重點是等,要給面試官介紹清楚。前言 在這互聯(lián)網(wǎng)的寒冬臘月時期,雖說過了金三銀四,但依舊在招人不斷。更偏向于招聘高級開發(fā)工程師。本人在這期間求職,去了幾家創(chuàng)業(yè),小廠,大廠廝殺了一番,也得到了自己滿意的offer。 整理一下自己還記得的面試...
摘要:學(xué)習(xí)開發(fā),無論是前端開發(fā)還是都避免不了要接觸異步編程這個問題就和其它大多數(shù)以多線程同步為主的編程語言不同的主要設(shè)計是單線程異步模型。由于異步編程可以實現(xiàn)非阻塞的調(diào)用效果,引入異步編程自然就是順理成章的事情了。 學(xué)習(xí)js開發(fā),無論是前端開發(fā)還是node.js,都避免不了要接觸異步編程這個問題,就和其它大多數(shù)以多線程同步為主的編程語言不同,js的主要設(shè)計是單線程異步模型。正因為js天生的與...
摘要:本周精讀內(nèi)容是逃離地獄。精讀仔細思考為什么會被濫用,筆者認為是它的功能比較反直覺導(dǎo)致的。同時,筆者認為,也不要過渡利用新特性修復(fù)新特性帶來的問題,這樣反而導(dǎo)致代碼可讀性下降。 本周精讀內(nèi)容是 《逃離 async/await 地獄》。 1 引言 終于,async/await 也被吐槽了。Aditya Agarwal 認為 async/await 語法讓我們陷入了新的麻煩之中。 其實,筆者...
摘要:當(dāng)然是否需要培訓(xùn)這個話題,得基于兩個方面,如果你是計算機專業(yè)畢業(yè)的,大學(xué)基礎(chǔ)課程學(xué)的還可以,我建議不需要去培訓(xùn),既然有一定的基礎(chǔ),那就把去培訓(xùn)浪費的四個月,用去實習(xí),培訓(xùn)是花錢,實習(xí)是掙錢,即使工資低點,一正一負自己算算吧。 上周一篇《程序員平時該如何學(xué)習(xí)來提高自己的技術(shù)》火了之后,「非著名程序員」微信公眾號的后臺經(jīng)常收到程序員和一些初學(xué)者的消息,問一些技術(shù)提高的問題,而且又恰逢畢業(yè)季...
閱讀 3428·2021-11-19 09:40
閱讀 1313·2021-10-11 11:07
閱讀 4844·2021-09-22 15:07
閱讀 2890·2021-09-02 15:15
閱讀 1964·2019-08-30 15:55
閱讀 539·2019-08-30 15:43
閱讀 883·2019-08-30 11:13
閱讀 1449·2019-08-29 15:36