上篇文章咱們簡單的介紹了nodejs中的事件event和事件循環event loop。本文本文將會更進一步,繼續講解nodejs中的event,並探討一下setTimeout,setImmediate和process.nextTick的區別。node
雖然nodejs是單線程的,可是nodejs能夠將操做委託給系統內核,系統內核在後臺處理這些任務,當任務完成以後,通知nodejs,從而觸發nodejs中的callback方法。linux
這些callback會被加入輪循隊列中,最終被執行。web
經過這樣的event loop設計,nodejs最終能夠實現非阻塞的IO。windows
nodejs中的event loop被分紅了一個個的phase,下圖列出了各個phase的執行順序:異步
每一個phase都會維護一個callback queue,這是一個FIFO的隊列。socket
當進入一個phase以後,首先會去執行該phase的任務,而後去執行屬於該phase的callback任務。函數
當這個callback隊列中的任務所有都被執行完畢或達到了最大的callback執行次數以後,就會進入下一個phase。oop
注意, windows和linux的具體實現有稍許不一樣,這裏咱們只關注最重要的幾個phase。
問題:phase的執行過程當中,爲何要限制最大的callback執行次數呢?ui
回答:在極端狀況下,某個phase可能會須要執行大量的callback,若是執行這些callback花費了太多的時間,那麼將會阻塞nodejs的運行,因此咱們設置callback執行的次數限制,以免nodejs的長時間block。this
上面的圖中,咱們列出了6個phase,接下來咱們將會一一的進行解釋。
timers的中文意思是定時器,也就是說在給定的時間或者時間間隔去執行某個callback函數。
一般的timers函數有這樣兩種:setTimeout和setInterval。
通常來講這些callback函數會在到期以後儘量的執行,可是會受到其餘callback執行的影響。 咱們來看一個例子:
const fs = require('fs'); function someAsyncOperation(callback) { // Assume this takes 95ms to complete fs.readFile('/path/to/file', callback); } const timeoutScheduled = Date.now(); setTimeout(() => { const delay = Date.now() - timeoutScheduled; console.log(`${delay}ms have passed since I was scheduled`); }, 100); // do someAsyncOperation which takes 95 ms to complete someAsyncOperation(() => { const startCallback = Date.now(); // do something that will take 10ms... while (Date.now() - startCallback < 10) { // do nothing } });
上面的例子中,咱們調用了someAsyncOperation,這個函數首先回去執行readFile方法,假設這個方法耗時95ms。接着執行readFile的callback函數,這個callback會執行10ms。最後纔回去執行setTimeout中的callback。
因此上面的例子中,雖然setTimeout指定要在100ms以後運行,可是實際上還要等待95 + 10 = 105 ms以後纔會真正的執行。
這個phase將會執行一些系統的callback操做,好比在作TCP鏈接的時候,TCP socket接收到了ECONNREFUSED信號,在某些liunx操做系統中將會上報這個錯誤,那麼這個系統的callback將會放到pending callbacks中運行。
或者是須要在下一個event loop中執行的I/O callback操做。
idle, prepare是內部使用的phase,這裏就不過多介紹。
poll將會檢測新的I/O事件,並執行與I / O相關的回調,注意這裏的回調指的是除了關閉callback,timers,和setImmediate以外的幾乎全部的callback事件。
poll主要處理兩件事情:輪詢I/O,而且計算block的時間,而後處理poll queue中的事件。
若是poll queue非空的話,event loop將會遍歷queue中的callback,而後一個一個的同步執行,知道queue消費完畢,或者達到了callback數量的限制。
由於queue中的callback是一個一個同步執行的,因此可能會出現阻塞的狀況。
若是poll queue空了,若是代碼中調用了setImmediate,那麼將會立馬跳到下一個check phase,而後執行setImmediate中的callback。 若是沒有調用setImmediate,那麼會繼續等待新來的callback被加入到queue中,並執行。
主要來執行setImmediate的callback。
setImmediate能夠看作是一個運行在單獨phase中的獨特的timer,底層使用的libuv API來規劃callbacks。
通常來講,若是在poll phase中有callback是以setImmediate的方式調用的話,會在poll queue爲空的狀況下,立馬結束poll phase,進入check phase來執行對應的callback方法。
最後一個phase是處理close事件中的callbacks。 好比一個socket忽然被關閉,那麼將會觸發一個close事件,並調用相關的callback。
setTimeout和setImmediate有什麼不一樣呢?
從上圖的phase階段能夠看出,setTimeout中的callback是在timer phase中執行的,而setImmediate是在check階段執行的。
從語義上講,setTimeout指的是,在給定的時間以後運行某個callback。而setImmediate是在執行完當前loop中的 I/O操做以後,立馬執行。
那麼這兩個方法的執行順序上有什麼區別呢?
下面咱們舉兩個例子,第一個例子中兩個方法都是在主模塊中運行:
setTimeout(() => { console.log('timeout'); }, 0); setImmediate(() => { console.log('immediate'); });
這樣運行兩個方法的執行順序是不肯定,由於可能受到其餘執行程序的影響。
第二個例子是在I/O模塊中運行這兩個方法:
const fs = require('fs'); fs.readFile(__filename, () => { setTimeout(() => { console.log('timeout'); }, 0); setImmediate(() => { console.log('immediate'); }); });
你會發現,在I/O模塊中,setImmediate必定會在setTimeout以前執行。
setTimeout和setImmediate二者都有一個返回值,咱們能夠經過這個返回值,來對timer進行clear操做:
const timeoutObj = setTimeout(() => { console.log('timeout beyond time'); }, 1500); const immediateObj = setImmediate(() => { console.log('immediately executing immediate'); }); const intervalObj = setInterval(() => { console.log('interviewing the interval'); }, 500); clearTimeout(timeoutObj); clearImmediate(immediateObj); clearInterval(intervalObj);
clear操做也能夠clear intervalObj。
setTimeout和setInterval返回的對象都是Timeout對象。
若是這個timeout對象是最後要執行的timeout對象,那麼可使用unref方法來取消其執行,取消執行完畢,可使用ref來恢復它的執行。
const timerObj = setTimeout(() => { console.log('will i run?'); }); timerObj.unref(); setImmediate(() => { timerObj.ref(); });
注意,若是有多個timeout對象,只有最後一個timeout對象的unref方法纔會生效。
process.nextTick也是一種異步API,可是它和timer是不一樣的。
若是咱們在一個phase中調用process.nextTick,那麼nextTick中的callback會在這個phase完成,進入event loop的下一個phase以前完成。
這樣作就會有一個問題,若是咱們在process.nextTick中進行遞歸調用的話,這個phase將會被阻塞,影響event loop的正常執行。
那麼,爲何咱們還會有process.nextTick呢?
考慮下面的一個例子:
let bar; function someAsyncApiCall(callback) { callback(); } someAsyncApiCall(() => { console.log('bar', bar); // undefined }); bar = 1;
上面的例子中,咱們定義了一個someAsyncApiCall方法,裏面執行了傳入的callback函數。
這個callback函數想要輸出bar的值,可是bar的值是在someAsyncApiCall方法以後被賦值的。
這個例子最終會致使輸出的bar值是undefined。
咱們的本意是想讓用戶程序執行完畢以後,再調用callback,那麼咱們可使用process.nextTick來對上面的例子進行改寫:
let bar; function someAsyncApiCall(callback) { process.nextTick(callback); } someAsyncApiCall(() => { console.log('bar', bar); // 1 }); bar = 1;
咱們再看一個實際中使用的例子:
const server = net.createServer(() => {}).listen(8080); server.on('listening', () => {});
上面的例子是最簡單的nodejs建立web服務。
上面的例子有什麼問題呢?listen(8000) 方法將會立馬綁定8000端口。可是這個時候,server的listening事件綁定代碼尚未執行。
這裏實際上就用到了process.nextTick技術,從而無論咱們在什麼地方綁定listening事件,均可以監聽到listen事件。
process.nextTick 是立馬在當前phase執行callback,而setImmediate是在check階段執行callback。
因此process.nextTick要比setImmediate的執行順序優先。
實際上,process.nextTick和setImmediate的語義應該進行互換。由於process.nextTick表示的纔是immediate,而setImmediate表示的是next tick。
本文做者:flydean程序那些事本文連接:http://www.flydean.com/nodejs-event-more/
本文來源:flydean的博客
歡迎關注個人公衆號:「程序那些事」最通俗的解讀,最深入的乾貨,最簡潔的教程,衆多你不知道的小技巧等你來發現!