摘要:事件循環(huán)背景是一門單線程非阻塞的腳本語言,單線程意味著,代碼在執(zhí)行的任何時候,都只有一個主線程來處理所有的任務(wù)。在意識到該問題之際,新特性中的可以讓成為一門多線程語言,但實際開發(fā)中使用存在著諸多限制。這個地方被稱為執(zhí)行棧。
事件循環(huán)(Event Loop) 背景
JavaScript是一門單線程非阻塞的腳本語言,單線程意味著,JavaScript代碼在執(zhí)行的任何時候,都只有一個主線程來處理所有的任務(wù)。而非阻塞則是當代碼需要進行一項異步任務(wù)(無法立刻返回結(jié)果,需要花一定時間才能返回的任務(wù),如I/O事件)的時候,主線程會掛起(pending)這個任務(wù),然后在異步任務(wù)返回結(jié)果的時候再根據(jù)一定規(guī)則去執(zhí)行相應(yīng)的回調(diào)。
在意識到該問題之際,html5新特性中的web worker可以讓JavaScript成為一門多線程語言,但實際開發(fā)中使用web worker存在著諸多限制。
思考一下這段代碼的結(jié)果?
顯然,同步函數(shù)是由上至下的執(zhí)行順序
因為setTimeout是異步函數(shù),js執(zhí)行機制是先將同步函數(shù)執(zhí)行完畢,再執(zhí)行異步函數(shù)
當javascript代碼執(zhí)行的時候會將不同的變量存于內(nèi)存中的不同位置:堆(heap)和棧(stack)中來加以區(qū)分。其中,堆里存放著一些對象。而棧中則存放著一些基礎(chǔ)類型變量以及對象的指針。如下:
當我們調(diào)用一個方法的時候,js會生成一個與這個方法對應(yīng)的執(zhí)行環(huán)境(context),又叫執(zhí)行上下文。這個執(zhí)行環(huán)境中存在著這個方法的私有作用域,上層作用域的指向,方法的參數(shù),這個作用域中定義的變量以及這個作用域的this對象。 而當一系列方法被依次調(diào)用的時候,因為js是單線程的,同一時間只能執(zhí)行一個方法,于是這些方法被排隊在一個多帶帶的地方。這個地方被稱為執(zhí)行棧。
當腳本第一次執(zhí)行的時候,js引擎會解析這段代碼,并將其中的同步代碼按照執(zhí)行順序加入執(zhí)行棧中,然后從頭開始執(zhí)行。如果當前執(zhí)行的是一個方法,那么js會向執(zhí)行棧中添加這個方法的執(zhí)行環(huán)境,然后進入這個執(zhí)行環(huán)境繼續(xù)執(zhí)行其中的代碼。當這個執(zhí)行環(huán)境中的代碼 執(zhí)行完畢并返回結(jié)果后,js會退出這個執(zhí)行環(huán)境并把這個執(zhí)行環(huán)境銷毀,回到上一個方法的執(zhí)行環(huán)境。這個過程反復(fù)進行,直到執(zhí)行棧中的代碼全部執(zhí)行完畢。
在異步函數(shù)中,可以細分為兩種任務(wù),宏任務(wù)與微任務(wù)。
宏任務(wù)有以下幾種:
①I/O
②setTimeout
③setInterval
④setImmediate
⑤requestAnimationFrame
微任務(wù)有以下幾種:
①process.nextTick
②MutationObserver
③Promise.then catch finally
當異步函數(shù)中同時存在微任務(wù)和宏任務(wù)的時候,先執(zhí)行完微任務(wù),再執(zhí)行宏任務(wù)
瀏覽器中的事件循環(huán)機制就是js在執(zhí)行代碼時,由上至下遍歷,優(yōu)先執(zhí)行同步函數(shù),在遇到異步函數(shù)的時候,將該任務(wù)放置執(zhí)行棧;當任務(wù)隊列中沒有同步函數(shù)之后便開始執(zhí)行執(zhí)行棧中的異步函數(shù),優(yōu)先執(zhí)行微任務(wù),后執(zhí)行宏任務(wù)。
js執(zhí)行順序: 同步函數(shù) -> 微任務(wù) -> 宏任務(wù)
console.log("script start") setTimeout(function(){ console.log("settimeout") }) console.log("script end") // 輸出順序:script start->script end->settimeout
這個很簡單,setTimeout中的語句會進入宏任務(wù),后置執(zhí)行。
promiseconsole.log("script start") let promise1 = new Promise(function (resolve) { console.log("promise1") resolve() console.log("promise1 end") }).then(function () { console.log("promise2") }) setTimeout(function(){ console.log("settimeout") }) console.log("script end") // 輸出順序: script start->promise1->promise1 end->script end->promise2->settimeout
兩個要點:
①promise中的then函數(shù)中代碼會進入微任務(wù)
②promise中的resolve只是更改promise的狀態(tài),因此后面的語句會繼續(xù)執(zhí)行。
async function async1(){ console.log("async1 start"); await async2(); console.log("async1 end") } async function async2(){ console.log("async2") } console.log("script start"); async1(); console.log("script end") // 輸出順序:script start->async1 start->async2->script end->async1 end
async函數(shù)表示函數(shù)里面可能會有異步方法,await后面跟一個表達式,async方法執(zhí)行時,遇到await會立即執(zhí)行表達式,然后把表達式后面的代碼放到微任務(wù)隊列里,讓出執(zhí)行棧讓同步代碼先執(zhí)行
綜合測試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");
這題不公布答案,自行答題。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://www.ezyhdfw.cn/yun/101946.html
摘要:如果沒有其他異步任務(wù)要處理比如到期的定時器,會一直停留在這個階段,等待請求返回結(jié)果。執(zhí)行的執(zhí)行事件關(guān)閉請求的,例如事件循環(huán)的每一次循環(huán)都需要依次經(jīng)過上述的階段。因此,才會早于執(zhí)行。 showImg(https://segmentfault.com/img/bVbnY76); 概念 同步任務(wù)(Synchronous) 在主線程上排隊執(zhí)行的任務(wù),只有前一個任務(wù)執(zhí)行完畢,才能執(zhí)行后一個任務(wù) ...
摘要:以多線程的形式,允許單個任務(wù)分成不同的部分進行運行。提供協(xié)調(diào)機制,一方面防止進程之間和線程之間產(chǎn)生沖突,另一方面允許進程之間和線程之間共享資源。主線程會不斷的重復(fù)上訴過程。 眾所周知,js是單線程的,說到線程,我們首先來仔細辨析一下線程和進程的知識。 一、進程與線程 阮一峰老師的一篇文章寫的很好 cpu會給當前進程分配資源,進程是資源分配的最小單位,進程的資源會分配給線程使用,線程是C...
摘要:關(guān)于循環(huán)和閉包當循環(huán)和閉包結(jié)合在一起時,經(jīng)常會產(chǎn)生讓初學(xué)者覺得匪夷所思的問題。閉包是一把雙刃劍是比較難以理解和掌握的部分,它十分強大,卻也有很大的缺陷,如何使用它完全取決于你自己。 在談閉包之前,我們首先要了解幾個概念: 什么是函數(shù)表達式? 與函數(shù)聲明有何不同? JavaScript查找標識符的機制 JavaScript的作用域是詞法作用域 JavaScript的垃圾回收機制 先來...
摘要:引言一直對瀏覽器的進程線程的運行一無所知,經(jīng)過一次的刷刷刷相關(guān)的博客之后,對其有了初步的了解,是時候該總結(jié)一波了。瀏覽器內(nèi)的進程知道了進程與線程之間的關(guān)系之后,下面是瀏覽器與進程的關(guān)系了。 引言 一直對瀏覽器的進程、線程的運行一無所知,經(jīng)過一次的刷刷刷相關(guān)的博客之后,對其有了初步的了解,是時候該總結(jié)一波了。 進程、線程之間的關(guān)系 一個進程有一個或多個線程,線程之間共同完成進程分配下來的...
摘要:事件簡介事件是合成事件,所有事件都自動綁定到最外層上。支持事件的冒泡機制,我們可以使用和來中斷它。這樣做簡化了事件處理和回收機制,效率也有很大提升。事件類型合成事件的事件類型是原生事件類型的一個子集。 React事件簡介 React事件是合成事件,所有事件都自動綁定到最外層上。因為Virtual DOM 在內(nèi)存中是以對象的形式存在的,所以React 基于 Virtual DOM 實現(xiàn)了...
閱讀 1955·2021-11-15 11:39
閱讀 1154·2020-12-03 17:06
閱讀 803·2019-12-27 11:42
閱讀 3320·2019-08-30 13:59
閱讀 1542·2019-08-26 13:22
閱讀 3327·2019-08-26 12:15
閱讀 2523·2019-08-26 10:22
閱讀 1639·2019-08-23 18:40