成人无码视频,亚洲精品久久久久av无码,午夜精品久久久久久毛片,亚洲 中文字幕 日韩 无码

資訊專欄INFORMATION COLUMN

初窺JavaScript事件機(jī)制的實(shí)現(xiàn)(二)—— Node.js中定時(shí)器的實(shí)現(xiàn)

novo / 3300人閱讀

摘要:下面通過(guò)幾個(gè)的定時(shí)器示例以及相關(guān)源碼來(lái)分析在中,功能到底是怎么實(shí)現(xiàn)的。我們知道,中的定時(shí)器并不同于計(jì)算機(jī)底層的定時(shí)中斷。補(bǔ)充資料在高級(jí)程序設(shè)計(jì)第三版第章高級(jí)技巧中對(duì)高級(jí)定時(shí)器以及有較詳細(xì)的討論。至此,這類定時(shí)器函數(shù)已經(jīng)可以為所用了。

上一篇博文提到,在Node中timer并不是通過(guò)新開線程來(lái)實(shí)現(xiàn)的,而是直接在event loop中完成。下面通過(guò)幾個(gè)JavaScript的定時(shí)器示例以及Node相關(guān)源碼來(lái)分析在Node中,timer功能到底是怎么實(shí)現(xiàn)的。

JavaScript中定時(shí)器功能的特點(diǎn)

無(wú)論是Node還是瀏覽器中,都有setTimeout和setInterval這兩個(gè)定時(shí)器函數(shù),并且其工作特點(diǎn)基本相同,因此下面僅以Node為例進(jìn)行分析。

我們知道,JavaScript中的定時(shí)器并不同于計(jì)算機(jī)底層的定時(shí)中斷。中斷到來(lái)時(shí),當(dāng)前執(zhí)行代碼會(huì)被打斷,轉(zhuǎn)去執(zhí)行定時(shí)中斷處理函數(shù)。而JavaScript的定時(shí)器到時(shí),如果當(dāng)前執(zhí)行線程沒(méi)有正在執(zhí)行的代碼,則執(zhí)行相應(yīng)的回調(diào)函數(shù);如果當(dāng)前有代碼在執(zhí)行中,JavaScript引擎既不會(huì)中斷當(dāng)前代碼轉(zhuǎn)去執(zhí)行回調(diào),也不會(huì)開新的線程執(zhí)行回調(diào),而是當(dāng)前代碼執(zhí)行完畢之后才去處理。

console.time("A")
setTimeout(function () {
    console.timeEnd("A");
}, 100);
var i = 0;
for (; i < 100000; i++) { }

執(zhí)行上面的代碼,可以看到最終輸出的時(shí)間并不是100ms左右,而是數(shù)秒。這說(shuō)明在循環(huán)完成之前,定時(shí)回調(diào)函數(shù)確實(shí)沒(méi)有被執(zhí)行,而是推遲到了循環(huán)結(jié)束。實(shí)際上在JavaScript代碼執(zhí)行中,所有的事件都無(wú)法得到處理,必須等到當(dāng)前代碼全部完成,才能去處理新的事件。這就是為什么在瀏覽器中運(yùn)行耗時(shí)JavaScript代碼時(shí),瀏覽器會(huì)失去響應(yīng)。為了應(yīng)對(duì)這種情況,我們可以采取Yielding Processes的技巧,將耗時(shí)的代碼分成小塊(chunks),每處理完一塊就執(zhí)行一次setTimeout,約定在一小段時(shí)間后才處理下一塊,而在這段空閑時(shí)間里,瀏覽器/Node可以去處理排隊(duì)中的事件。

補(bǔ)充資料

JavaScript 高級(jí)程序設(shè)計(jì) 第三版第22章高級(jí)技巧中對(duì)高級(jí)定時(shí)器以及Yielding Processes有較詳細(xì)的討論。

Node中的timer實(shí)現(xiàn) libuv對(duì)uv_loop_t類型的初始化

上一篇博文提到Node會(huì)調(diào)用libuv的uv_run函數(shù)啟動(dòng)default_loop_ptr進(jìn)行事件調(diào)度,default_loop_ptr指向一個(gè)uv_loop_t類型的變量default_loop_struct。Node啟動(dòng)時(shí)會(huì)調(diào)用uv_loop_init(&default_loop_struct)對(duì)其進(jìn)行初始化,uv_loop_init函數(shù)節(jié)選如下:

int uv_loop_init(uv_loop_t* loop) {
  ...
  loop->time = 0;
  uv_update_time(loop);
  ...
}

可以看到looptime字段先被賦值為0,之后調(diào)用uv_update_time函數(shù),這會(huì)將最新的計(jì)數(shù)時(shí)間賦給loop.time。

初始化完成之后,default_loop_struct.time就有了一個(gè)初始值,與時(shí)間有關(guān)的操作都會(huì)與此值進(jìn)行比較從而確定是否調(diào)用相應(yīng)回調(diào)函數(shù)。

libuv的事件調(diào)度核心

前面提到uv_run函數(shù)就是libuv庫(kù)實(shí)現(xiàn)event loop的核心部分,下面是其流程圖:

這里簡(jiǎn)述一下上面與定時(shí)器相關(guān)的邏輯:

更新當(dāng)前looptime字段,這個(gè)字段標(biāo)志著當(dāng)前loop概念下的“現(xiàn)在”;

檢查loop是否alive,也就是說(shuō)檢查loop中是否還有需要處理的任務(wù)(handlers/requests),如果沒(méi)有就不必循環(huán)了;

檢查注冊(cè)過(guò)的timer,如果某一個(gè)timer中指定的時(shí)間落后于當(dāng)前時(shí)間了,說(shuō)明該timer已到時(shí),于是執(zhí)行其對(duì)應(yīng)的回調(diào)函數(shù);

執(zhí)行一次I/O polling(即阻塞住線程,等待I/O事件發(fā)生),如果在下一個(gè)timer到期時(shí)還沒(méi)有任何I/O完成,則停止等待,執(zhí)行下一個(gè)timer的回調(diào)。
如果發(fā)生了I/O事件,則執(zhí)行對(duì)應(yīng)的回調(diào);由于執(zhí)行回調(diào)的時(shí)間里可能又有timer到期了,這里要再次檢查timer并執(zhí)行回調(diào)。
(實(shí)際上(4.)這里比較復(fù)雜,不僅僅是一步操作,這樣描述僅是為了不涉及其他細(xì)節(jié),而專注于timer的實(shí)現(xiàn)。)

Node會(huì)一直調(diào)用uv_run直到loop不再alive。

Node中的timer_wrap與timers

Node中有一個(gè)TimerWrap類,被注冊(cè)為Node內(nèi)部的timer_wrap模塊。

NODE_MODULE_CONTEXT_AWARE_BUILTIN(timer_wrap, node::TimerWrap::Initialize)

其中TimerWrap類基本上就是對(duì)uv_timer_t的一個(gè)直接封裝,NODE_MODULE_CONTEXT_AWARE_BUILTIN是Node用于注冊(cè)built-in模塊的宏。

經(jīng)過(guò)這一步操作,JavaScript就可以拿到這個(gè)模塊進(jìn)行操作了。src/lib/timers.js文件使用JavaScript的形式把timer_wrap的功能封裝起來(lái),并導(dǎo)出了exports.setTimeout, exports.setInterval, exports.setImmediate等函數(shù)。

Node啟動(dòng)與global初始化

上一篇提到Node啟動(dòng)時(shí)會(huì)載入執(zhí)行環(huán)境LoadEnvironment(env),這個(gè)函數(shù)中非常重要的一步就是載入src/node.js并執(zhí)行,src/node.js會(huì)載入指定的模塊并初始化globalprocess。當(dāng)然,setTimeout等函數(shù)也會(huì)被src/node.js綁定到global對(duì)象上。

至此,setTimeout/setInterval這類定時(shí)器函數(shù)已經(jīng)可以為JavaScript所用了。

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

轉(zhuǎn)載請(qǐng)注明本文地址:http://m.hztianpu.com/yun/85764.html

相關(guān)文章

  • 初窺JavaScript事件機(jī)制實(shí)現(xiàn)(一)—— Node.js事件驅(qū)動(dòng)實(shí)現(xiàn)概覽

    摘要:如果當(dāng)前沒(méi)有事件也沒(méi)有定時(shí)器事件,則返回。相關(guān)資料關(guān)于的架構(gòu)及設(shè)計(jì)思路的事件討論了使用線程池異步運(yùn)行代碼。下一篇初窺事件機(jī)制的實(shí)現(xiàn)二中定時(shí)器的實(shí)現(xiàn) 在瀏覽器中,事件作為一個(gè)極為重要的機(jī)制,給予JavaScript響應(yīng)用戶操作與DOM變化的能力;在Node.js中,事件驅(qū)動(dòng)模型則是其高并發(fā)能力的基礎(chǔ)。 學(xué)習(xí)JavaScript也需要了解它的運(yùn)行平臺(tái),為了更好的理解JavaScript的事...

    lavor 評(píng)論0 收藏0
  • Node_深入淺出Node

    摘要:簡(jiǎn)介項(xiàng)目命名為就是一個(gè)服務(wù)器單純開發(fā)一個(gè)服務(wù)器的想法,變成構(gòu)建網(wǎng)絡(luò)應(yīng)用的一個(gè)基本框架發(fā)展為一個(gè)強(qiáng)制不共享任何資源的單線程,單進(jìn)程系統(tǒng)。單線程弱點(diǎn)無(wú)法利用多核錯(cuò)誤會(huì)引起整個(gè)應(yīng)用退出,應(yīng)用的健壯性大量計(jì)算占用導(dǎo)致無(wú)法繼續(xù)調(diào)用異步。 NodeJs簡(jiǎn)介 Ryan Dahl項(xiàng)目命名為:web.js 就是一個(gè)Web服務(wù)器.單純開發(fā)一個(gè)Web服務(wù)器的想法,變成構(gòu)建網(wǎng)絡(luò)應(yīng)用的一個(gè)基本框架.Node發(fā)展...

    shinezejian 評(píng)論0 收藏0
  • JavaScript執(zhí)行機(jī)制

    摘要:事件循環(huán)事件循環(huán)是實(shí)現(xiàn)異步的一種方法,也是的執(zhí)行機(jī)制。最后的最后是一門單線程語(yǔ)言是的執(zhí)行機(jī)制部分內(nèi)容轉(zhuǎn)自 1.單線程 javascript是一門單線程語(yǔ)言 2.javascript事件循環(huán) 同步任務(wù) 異步任務(wù) showImg(https://segmentfault.com/img/bVbufUd?w=1268&h=1062);除了廣義的同步任務(wù)和異步任務(wù),我們對(duì)任務(wù)有更精細(xì)的定義...

    ralap 評(píng)論0 收藏0
  • 事件循環(huán)與任務(wù)隊(duì)列

    摘要:需要注意的是,定時(shí)器比較特殊,并沒(méi)有把回調(diào)函數(shù)掛在事件循環(huán)隊(duì)列中,它所做的就是設(shè)置一個(gè)定時(shí)器,當(dāng)定時(shí)器到時(shí)后,環(huán)境會(huì)把你的回調(diào)函數(shù)放在事件循環(huán)中,這樣,在未來(lái)某個(gè)時(shí)刻的會(huì)被取出執(zhí)行。 Author: bugall Wechat: bugallF Email: 769088641@qq.com Github: https://github.com/bugall 一...

    SQC 評(píng)論0 收藏0
  • 瀏覽器與Node事件循環(huán)(Event Loop)有何區(qū)別?

    摘要:事件觸發(fā)線程主要負(fù)責(zé)將準(zhǔn)備好的事件交給引擎線程執(zhí)行。它將不同的任務(wù)分配給不同的線程,形成一個(gè)事件循環(huán),以異步的方式將任務(wù)的執(zhí)行結(jié)果返回給引擎。 Fundebug經(jīng)作者浪里行舟授權(quán)首發(fā),未經(jīng)同意請(qǐng)勿轉(zhuǎn)載。 前言 本文我們將會(huì)介紹 JS 實(shí)現(xiàn)異步的原理,并且了解了在瀏覽器和 Node 中 Event Loop 其實(shí)是不相同的。 一、線程與進(jìn)程 1. 概念 我們經(jīng)常說(shuō) JS 是單線程執(zhí)行的,...

    TANKING 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

閱讀需要支付1元查看
<