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

資訊專欄INFORMATION COLUMN

js_Event Loop(筆記)

liaoyg8023 / 728人閱讀

摘要:單線程的好處簡單,處理時不會出現(xiàn)并發(fā)競爭問題異步的必要性讓用戶體驗更流暢如何實現(xiàn)異步見參考,,調(diào)用棧函數(shù)執(zhí)行上下文。單線程只能有一個并且每次只能執(zhí)行一個任務。

參考:

JavaScript 運行機制詳解:再談Event Loop

深入理解JavaScript的執(zhí)行過程--單線程的JS

細說JavaScript單線程的一些事

The JavaScript Event Loop: Explained

模擬Event Loop執(zhí)行過程

Node.js 事件循環(huán)一: 淺析

理解 Event Loop、Micro Task & Macro Task

HTML系列:macrotask和microtask

1. js為啥是單線程執(zhí)行

單線程是指只在一個線程里執(zhí)行JS代碼,但瀏覽器是多線程的。

1.1 單線程的好處

簡單,處理DOM時不會出現(xiàn)并發(fā)競爭問題

2. 異步的必要性

讓用戶體驗更流暢

3. 如何實現(xiàn)異步:Event Loop

見參考1,4,5

3.1 調(diào)用棧(call Stack)

函數(shù)執(zhí)行上下文??刂茍?zhí)行函數(shù)的調(diào)用執(zhí)行。單線程只能有一個call statck, 并且每次只能執(zhí)行一個任務。棧是有尺寸的,即在一次執(zhí)行中函數(shù)的嵌套調(diào)用數(shù)量是有限的,如果超過這個限制就會報錯。JS執(zhí)行時遇到這種錯誤"Uncaught RangeError: Maximum call stack size exceeded"就是說明call statck溢出了。回調(diào)函數(shù)上限數(shù)量取決于statck本身的大小以及statck元素的大小->見參考:

function computeMaxCallStackSize1() {
    try {
        return 1 + computeMaxCallStackSize1();
    } catch (e) {
        // Call stack overflow
        return 1;
    }
}

// 多了形參p,call stack元素就多占了內(nèi)存
function computeMaxCallStackSize2(p) {
    try {
        return 1 + computeMaxCallStackSize2();
    } catch (e) {
        // Call stack overflow
        return 1;
    }
}

console.log(computeMaxCallStackSize1()); // 兩個輸出結(jié)果不一樣
console.log(computeMaxCallStackSize2());
3.2 回調(diào)隊列(callback queue)

看參考圖,生動的動畫模擬可以見參考5。當call stack空的時候,主線程查看callback隊列里是否有異步任務,如果有則取出執(zhí)行,執(zhí)行完后(即call stack變空了)再去查看call back隊列是否有異步任務。這個是循環(huán)的過程,也叫事件循環(huán)(event loop)。

while (queue.waitForMessage()) {
  queue.processNextMessage();
}

個人認為事件隊列主要實現(xiàn)了異步回調(diào)功能。

3.3 非阻塞I/O

首先明確一點,JS本身就沒有I/O(網(wǎng)絡請求,磁盤讀寫,用戶交互)的,所有的I/O操作都是由宿主執(zhí)行的,宿主提供相關的API供JS調(diào)用。當JS調(diào)用IO API時并不會等待宿主的執(zhí)行結(jié)果而是繼續(xù)執(zhí)行后面的代碼,當IO執(zhí)行完成后宿主再通知JS,即在eventQueue中插入回調(diào)task。
不過還有些I/O是同步的,比如同步XHR請求,alert。盡量避免使用。

3.4 阻塞eventLoop

event loop實現(xiàn)了異步回調(diào),但回調(diào)只能一個一個的執(zhí)行,但前面的一個回調(diào)非常耗時時就會阻塞后面的回調(diào)執(zhí)行,用戶交互也可能出現(xiàn)卡死。

        function sleep(ms) {
            var curr = Date.now();
            while(Date.now() - curr < ms){};
        }
        console.log(1)
        setTimeout(function(){ // 回調(diào)依舊被阻塞
            console.log(2) 
        }, 0)
        sleep(2000)

最好開啟新線程執(zhí)行耗時的運算(使用web worker)或者放在服務端運算。

4. Macro-task & micro-task

event loop中除了 callback queue(macro-task)外還有個隊列專門處理micro-task。micor-task隊列為空時才去處理macro-task隊列執(zhí)行。
當打算以同步的方式處理異步回調(diào)(即執(zhí)行棧為空時立馬執(zhí)行回調(diào)函數(shù))時可以采用micro-task方式。

macrotasks: setTimeout, setInterval, setImmediate, requestAnimationFrame, I/O, UI rendering

microtasks: process.nextTick, Promises, Object.observe, MutationObserver

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

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

相關文章

  • 06.Android之消息機制問題

    摘要:通過向消息池發(fā)送各種消息事件通過處理相應的消息事件。子線程往消息隊列發(fā)送消息,并且往管道文件寫數(shù)據(jù),主線程即被喚醒,從管道文件讀取數(shù)據(jù),主線程被喚醒只是為了讀取消息,當消息讀取完畢,再次睡眠。 目錄介紹 6.0.0.1 談談消息機制Hander作用?有哪些要素?流程是怎樣的? 6.0.0.2 為什么一個線程只有一個Looper、只有一個MessageQueue,可以有多個Handle...

    Aomine 評論0 收藏0

發(fā)表評論

0條評論

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