成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

總結javascript基礎概念(二):事件隊列循環(huán)

zhkai / 1654人閱讀

摘要:而事件循環(huán)是主線程中執(zhí)行棧里的代碼執(zhí)行完畢之后,才開始執(zhí)行的。由此產(chǎn)生的異步事件執(zhí)行會作為任務隊列掛在當前循環(huán)的末尾執(zhí)行。在下,觀察者基于監(jiān)聽事件的完成情況在下基于多線程創(chuàng)建。

主要問題:

1、JS引擎是單線程,如何完成事件循環(huán)的?
2、定時器函數(shù)為什么計時不準確?
3、回調(diào)與異步,有什么聯(lián)系和不同?
4、ES6的事件循環(huán)有什么變化?Node中呢?
5、異步控制有什么難點?有什么解決方案?

二、事件隊列循環(huán)

(一)、瀏覽器線程

JavaScript引擎是基于事件驅(qū)動單線程運行的,瀏覽器無論在什么時候都只且只有一個線程在運行JavaScript程序,等待著任務隊列中任務的到來,然后加以處理。

瀏覽器的內(nèi)核是多線程的,它們在內(nèi)核控制下相互配合協(xié)作,一個瀏覽器至少實現(xiàn)三個常駐線程:JavaScript引擎線程,GUI渲染線程,瀏覽器事件線程。

GUI渲染線程負責渲染瀏覽器界面,當界面需要重繪(Repaint)或由于某種操作引發(fā)回流(Reflow)時,該線程就會執(zhí)行。GUI渲染線程與JavaScript引擎是互斥的,當JavaScript引擎執(zhí)行時GUI線程會被掛起,GUI更新會被保存在一個隊列中等到JavaScript引擎空閑時立即被執(zhí)行。所以渲染操作的消耗特別大。

(一)、事件類型與隊列
事件循環(huán):引擎會創(chuàng)建一個類似于 while (true) 的無限循環(huán),每執(zhí)行一次循環(huán)體的過程稱之為 Tick。每次 Tick 的過程就是查看是否有待處理事件,如果有則取出相關事件及回調(diào)函數(shù)放入執(zhí)行棧中由主線程執(zhí)行。待處理的事件會存儲在一個任務隊列中,也就是每次 Tick 會查看任務隊列中是否有需要執(zhí)行的任務。

任務隊列:異步操作會將相關回調(diào)添加到任務隊列中。而不同的異步操作添加到任務隊列的時機也不同,如 onclick, setTimeout, ajax 處理的方式都不同,這些異步操作是由瀏覽器內(nèi)核的 webcore 來執(zhí)行的,webcore 包含上圖中的3種 webAPI,分別是 DOM Binding、network、timer模塊。

onclick 由瀏覽器內(nèi)核的 DOM Binding 模塊來處理,當事件觸發(fā)的時候,回調(diào)函數(shù)會立即添加到任務隊列中。
setTimeout 會由瀏覽器內(nèi)核的 timer 模塊來進行延時處理,當時間到達的時候,才會將回調(diào)函數(shù)添加到任務隊列中。

js定時器不準確的原因:
1、異步函數(shù)的回調(diào)執(zhí)行會阻塞下一個循環(huán)tick。2、瀏覽器的時間精度不一。

ajax 則會由瀏覽器內(nèi)核的 network 模塊來處理,在網(wǎng)絡請求完成返回之后,才將回調(diào)添加到任務隊列中。

主線程:JS 只有一個線程,稱之為主線程。而事件循環(huán)是主線程中執(zhí)行棧里的代碼執(zhí)行完畢之后,才開始執(zhí)行的。所以,主線程中要執(zhí)行的代碼時間過長,會阻塞事件循環(huán)的執(zhí)行,也就會阻塞異步操作的執(zhí)行。只有當主線程中執(zhí)行棧為空的時候(即同步代碼執(zhí)行完后),才會進行事件循環(huán)來觀察要執(zhí)行的事件回調(diào),當事件循環(huán)檢測到任務隊列中有事件就取出相關回調(diào)放入執(zhí)行棧中由主線程執(zhí)行。

在這個過程提高頁面響應速度的方式:減少主線程同步代碼的數(shù)量,
將不重要的代碼轉(zhuǎn)移到事件循環(huán)階段執(zhí)行

(三)、ES6的任務隊列
對事件隊列進行了改造,使得異步回調(diào)可以更早的執(zhí)行,每個tick間隙都會優(yōu)先執(zhí)行任務隊列,不用排到事件隊列的末尾。一道必考題,看一下執(zhí)行順序:


Promise實例具有三種狀態(tài):等待,決議,拒絕。由此產(chǎn)生的異步事件執(zhí)行會作為任務隊列掛在當前tick循環(huán)的末尾執(zhí)行。

(四)、Nodo中的事件循環(huán)
JS引擎的事件循環(huán)都需要宿主環(huán)境提供隊列維護,Node脫離了瀏覽器,用到了不同的方式和底層系統(tǒng)做交互。

觀察者:引擎在每個循環(huán)過程中詢問觀察者是否有要處理的事件。

在Window下,觀察者基于IOCP監(jiān)聽事件的完成情況;在*nix下基于多線程創(chuàng)建。

在node增加了異步執(zhí)行的api,如process。nextTick()和setImmiediate。
process。nextTick()的回調(diào)函數(shù)保存在一個數(shù)組中,setImmiediate則是保存在一個鏈表中。
在每輪循環(huán)中,會將nextTick的數(shù)組中回調(diào)函數(shù)全部執(zhí)行完,然后執(zhí)行一個setImmiediate鏈表中的回調(diào)。

(五)、異步事件的處理
異步事件的處理過程中有執(zhí)行時間、順序不確定,回調(diào)地獄,錯誤難捕獲定位等問題。

1、jQuery的 Deferred隊列模塊
JQ在 Deferred隊列模塊的基礎上,模擬實現(xiàn)了promise相似的方法??梢宰龅芥準交卣{(diào),延遲訪問。
2、ES6 Promise
promise實例決議后就不可更改,解決了信任問題??梢允褂面準交卣{(diào),隨時訪問異步事件的狀態(tài)。相比之前要將回調(diào)交給另一方控制,promise可以將回調(diào)的執(zhí)行控制在自己的邏輯中。至此仍未從實質(zhì)上解決異步。
3、Generator
生成器函數(shù)實現(xiàn)了真正的異步控制,可以切換執(zhí)行環(huán)境,并在執(zhí)行環(huán)境之間傳遞變量,實現(xiàn)協(xié)作的函數(shù)線程。
著名的co模塊,結合使用Promise和Generator實現(xiàn)了異步流程的同步編碼形式。
4、async 函數(shù)
ES2017 標準引入了 async 函數(shù),使得異步操作變得更加方便。就是 Generator 函數(shù)的語法糖。
Node框架Koa采用了最新的async處理異步流程,使得編碼更加簡潔流暢。

推薦參考書:

《深入淺出NodeJS》
《jQuery技術內(nèi)幕》
《Webkit技術內(nèi)幕》
《你不知道的JavaScript》

文章版權歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/107393.html

相關文章

  • JavaScript:徹底理解同步、異步和事件循環(huán)(Event Loop)

    摘要:例如處理請求的線程處理事件的線程定時器線程讀寫文件的線程例如在中等等。事件循環(huán)事件循環(huán)是指主線程重復從消息隊列中取消息執(zhí)行的過程。事件觸發(fā)時,表示異步任務完成,會將事件監(jiān)聽器函數(shù)封裝成一條消息放到消息隊列中,等待主線程執(zhí)行。 一. 單線程 我們常說JavaScript是單線程的。 所謂單線程,是指在JS引擎中負責解釋和執(zhí)行JavaScript代碼的線程只有一個。不妨叫它主線程。 但是實...

    wenyiweb 評論0 收藏0
  • JS異步那些事 一 (基礎知識)

    摘要:異步那些事一基礎知識異步那些事二分布式事件異步那些事三異步那些事四異步那些事五異步腳本加載事件概念異步回調(diào)首先了講講中兩個方法和定義和用法方法用于在指定的毫秒數(shù)后調(diào)用函數(shù)或計算表達式。功能在事件循環(huán)的下一次循環(huán)中調(diào)用回調(diào)函數(shù)。 JS異步那些事 一 (基礎知識)JS異步那些事 二 (分布式事件)JS異步那些事 三 (Promise)JS異步那些事 四(HTML 5 Web Workers...

    李濤 評論0 收藏0
  • javascript引擎執(zhí)行的過程的理解--執(zhí)行階段

    摘要:如果對語法分析和預編譯,還有疑問引擎執(zhí)行的過程的理解語法分析和預編譯階段。參與執(zhí)行過程的線程分別是引擎線程也稱為內(nèi)核,負責解析執(zhí)行腳本程序的主線程例如引擎。以上便是引擎執(zhí)行宏任務的整個過程。 一、概述 js引擎執(zhí)行過程主要分為三個階段,分別是語法分析,預編譯和執(zhí)行階段,上篇文章我們介紹了語法分析和預編譯階段,那么我們先做個簡單概括,如下: 1、語法分析: 分別對加載完成的代碼塊進行語法...

    SnaiLiu 評論0 收藏0
  • javascript引擎執(zhí)行的過程的理解--執(zhí)行階段

    摘要:如果對語法分析和預編譯,還有疑問引擎執(zhí)行的過程的理解語法分析和預編譯階段。參與執(zhí)行過程的線程分別是引擎線程也稱為內(nèi)核,負責解析執(zhí)行腳本程序的主線程例如引擎。以上便是引擎執(zhí)行宏任務的整個過程。一、概述 js引擎執(zhí)行過程主要分為三個階段,分別是語法分析,預編譯和執(zhí)行階段,上篇文章我們介紹了語法分析和預編譯階段,那么我們先做個簡單概括,如下: 1、語法分析: 分別對加載完成的代碼塊進行語法檢驗,語...

    Achilles 評論0 收藏0
  • 前端基礎進階(一):內(nèi)存空間詳細圖解

    摘要:一棧數(shù)據(jù)結構與不同,中并沒有嚴格意義上區(qū)分棧內(nèi)存與堆內(nèi)存。引用數(shù)據(jù)類型的值是保存在堆內(nèi)存中的對象。不允許直接訪問堆內(nèi)存中的位置,因此我們不能直接操作對象的堆內(nèi)存空間。為了更好的搞懂變量對象與堆內(nèi)存,我們可以結合以下例子與圖解進行理解。 showImg(https://segmentfault.com/img/remote/1460000009784102?w=1240&h=683); ...

    _Suqin 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<