摘要:異步和事件驅(qū)動注本文是對眾多博客的學習和總結(jié),可能存在理解錯誤。接觸有兩個月,對的兩大特性一直有點模糊,即異步和事件驅(qū)動。
nodejs 異步I/O和事件驅(qū)動
注:本文是對眾多博客的學習和總結(jié),可能存在理解錯誤。請帶著懷疑的眼光,同時如果有錯誤希望能指出。
接觸nodejs有兩個月,對nodejs的兩大特性一直有點模糊,即異步IO和事件驅(qū)動。通過對《深入淺出nodejs》和幾篇博客的閱讀以后,有了大致的了解,總結(jié)一下。
幾個例子在開始之前,先來看幾個簡單例子,這也是我在使用nodejs時候遇到的幾個比較困惑的例子。
example 1var fs = require("fs"); var debug = require("debug")("example1"); debug("begin"); setTimeout(function(){ debug("timeout1"); }); setTimeout(function(){ debug("timeout2"); }); debug("end"); /** 運行結(jié)果 Sat, 21 May 2016 08:41:09 GMT example1 begin Sat, 21 May 2016 08:41:09 GMT example1 end Sat, 21 May 2016 08:41:09 GMT example1 timeout1 Sat, 21 May 2016 08:41:09 GMT example1 timeout2 */
question 1
example 2為何timeout1和timeout2的結(jié)果會在end后面?
var fs = require("fs"); var debug = require("debug")("example2"); debug("begin"); setTimeout(function(){ debug("timeout1"); }); setTimeout(function(){ debug("timeout2"); }); debug("end"); while(true); /** 運行結(jié)果 Sat, 21 May 2016 08:45:47 GMT example2 begin Sat, 21 May 2016 08:45:47 GMT example2 end */
question 2
example 3為何timeout1和timeout2沒有輸出到終端?while(true)到底阻塞了什么?
var fs = require("fs"); var debug = require("debug")("example3"); debug("begin"); setTimeout(function(){ debug("timeout1"); while (true); }); setTimeout(function(){ debug("timeout2"); }); debug("end"); /** 運行結(jié)果 Sat, 21 May 2016 08:49:12 GMT example3 begin Sat, 21 May 2016 08:49:12 GMT example3 end Sat, 21 May 2016 08:49:12 GMT example3 timeout1 */
question 3
example 4為什么timeout1中回調(diào)函數(shù)會阻塞timeout2中的回調(diào)函數(shù)的執(zhí)行?
var fs = require("fs"); var debug = require("debug")("example4"); debug("begin"); setTimeout(function(){ debug("timeout1"); /** * 模擬計算密集 */ for(var i = 0 ; i < 1000000 ; ++i){ for(var j = 0 ; j < 100000 ; ++j); } }); setTimeout(function(){ debug("timeout2"); }); debug("end"); /** Sat, 21 May 2016 08:53:27 GMT example4 begin Sat, 21 May 2016 08:53:27 GMT example4 end Sat, 21 May 2016 08:53:27 GMT example4 timeout1 Sat, 21 May 2016 08:54:09 GMT example4 timeout2 //注意這里的時間晚了好久 */
question 4
example 5和上面的問題一樣,為何timeout1的計算密集型工作將會阻塞timeout2的回調(diào)函數(shù)的執(zhí)行?
var fs = require("fs"); var debug = require("debug")("example5"); debug("begin"); fs.readFile("package.json","utf-8",function(err,data){ if(err) debug(err); else debug("get file content"); }); setTimeout(function(){ debug("timeout2"); }); debug("end"); /** 運行結(jié)果 Sat, 21 May 2016 08:59:14 GMT example5 begin Sat, 21 May 2016 08:59:14 GMT example5 end Sat, 21 May 2016 08:59:14 GMT example5 timeout2 Sat, 21 May 2016 08:59:14 GMT example5 get file content */
question 5
為何讀取文件的IO操作不會阻塞timeout2的執(zhí)行?
接下來我們就帶著上面幾個疑惑去理解nodejs中的異步IO和事件驅(qū)動是如何工作的。
異步IO(asynchronous I/O)首先來理解幾個容易混淆的概念,阻塞IO(blocking I/O)和非阻塞IO(non-blocking I/O),同步IO(synchronous I/O)和異步IO(synchronous I/O)。
博主一直天真的以為非阻塞I/O就是異步I/O T_T,apue一直沒有讀懂。
阻塞I/O 和 非阻塞I/O簡單來說,阻塞I/O就是當用戶發(fā)一個讀取文件描述符的操作的時候,進程就會被阻塞,直到要讀取的數(shù)據(jù)全部準備好返回給用戶,這時候進程才會解除block的狀態(tài)。
那非阻塞I/O呢,就與上面的情況相反,用戶發(fā)起一個讀取文件描述符操作的時,函數(shù)立即返回,不作任何等待,進程繼續(xù)執(zhí)行。但是程序如何知道要讀取的數(shù)據(jù)已經(jīng)準備好了呢?最簡單的方法就是輪詢。
除此之外,還有一種叫做IO多路復用的模式,就是用一個阻塞函數(shù)同時監(jiān)聽多個文件描述符,當其中有一個文件描述符準備好了,就馬上返回,在linux下,select,poll,epoll都提供了IO多路復用的功能。
同步I/O 和 異步I/O那么同步I/O和異步I/O又有什么區(qū)別么?是不是只要做到非阻塞IO就可以實現(xiàn)異步I/O呢?
其實不然。
同步I/O(synchronous I/O)做I/O operation的時候會將process阻塞,所以阻塞I/O,非阻塞I/O,IO多路復用I/O都是同步I/O。
異步I/O(asynchronous I/O)做I/O opertaion的時候?qū)⒉粫斐扇魏蔚淖枞?/p>
非阻塞I/O都不阻塞了為什么不是異步I/O呢?其實當非阻塞I/O準備好數(shù)據(jù)以后還是要阻塞住進程去內(nèi)核拿數(shù)據(jù)的。所以算不上異步I/O。
這里借一張圖(圖來自這里)來說明他們之間的區(qū)別
][1]
更多IO更多的詳細內(nèi)容可以在這里找到:
Linux IO模式及 select、poll、epoll詳解
select / poll / epoll: practical difference for system architects
事件驅(qū)動事件驅(qū)動(event-driven)是nodejs中的第二大特性。何為事件驅(qū)動呢?簡單來說,就是通過監(jiān)聽事件的狀態(tài)變化來做出相應的操作。比如讀取一個文件,文件讀取完畢,或者文件讀取錯誤,那么就觸發(fā)對應的狀態(tài),然后調(diào)用對應的回掉函數(shù)來進行處理。
線程驅(qū)動和事件驅(qū)動那么線程驅(qū)動編程和事件驅(qū)動編程之間的區(qū)別是什么呢?
線程驅(qū)動就是當收到一個請求的時候,將會為該請求開一個新的線程來處理請求。一般存在一個線程池,線程池中有空閑的線程,會從線程池中拿取線程來進行處理,如果線程池中沒有空閑的線程,新來的請求將會進入隊列排隊,直到線程池中空閑線程。
事件驅(qū)動就是當進來一個新的請求的時,請求將會被壓入隊列中,然后通過一個循環(huán)來檢測隊列中的事件狀態(tài)變化,如果檢測到有狀態(tài)變化的事件,那么就執(zhí)行該事件對應的處理代碼,一般都是回調(diào)函數(shù)。
對于事件驅(qū)動編程來說,如果某個時間的回調(diào)函數(shù)是計算密集型,或者是阻塞I/O,那么這個回調(diào)函數(shù)將會阻塞后面所有事件回調(diào)函數(shù)的執(zhí)行。這一點尤為重要。
nodejs的事件驅(qū)動和異步I/O 事件驅(qū)動模型上面介紹了那么多的概念,現(xiàn)在我們來看看nodejs中的事件驅(qū)動和異步I/O是如何實現(xiàn)的.
nodejs是單線程(single thread)運行的,通過一個事件循環(huán)(event-loop)來循環(huán)取出消息隊列(event-queue)中的消息進行處理,處理過程基本上就是去調(diào)用該消息對應的回調(diào)函數(shù)。消息隊列就是當一個事件狀態(tài)發(fā)生變化時,就將一個消息壓入隊列中。
nodejs的時間驅(qū)動模型一般要注意下面幾個點:
因為是單線程的,所以當順序執(zhí)行js文件中的代碼的時候,事件循環(huán)是被暫停的。
當js文件執(zhí)行完以后,事件循環(huán)開始運行,并從消息隊列中取出消息,開始執(zhí)行回調(diào)函數(shù)
因為是單線程的,所以當回調(diào)函數(shù)被執(zhí)行的時候,事件循環(huán)是被暫停的
當涉及到I/O操作的時候,nodejs會開一個獨立的線程來進行異步I/O操作,操作結(jié)束以后將消息壓入消息隊列。
下面我們從一個簡單的js文件入手,來看看 nodejs是如何執(zhí)行的。
var fs = require("fs"); var debug = require("debug")("example1"); debug("begin"); fs.readFile("package.json","utf-8",function(err,data){ if(err) debug(err); else debug("get file content"); }); setTimeout(function(){ debug("timeout2"); }); debug("end"); // 運行到這里之前,事件循環(huán)是暫停的
同步執(zhí)行debug("begin")
異步調(diào)用fs.readFile(),此時會開一個新的線程去進行異步I/O操作
異步調(diào)用setTimeout(),馬上將超時信息壓入到消息隊列中
同步調(diào)用debug("end")
開啟事件循環(huán),彈出消息隊列中的信息(目前是超時信息)
然后執(zhí)行信息對應的回調(diào)函數(shù)(事件循環(huán)又被暫停)
回調(diào)函數(shù)執(zhí)行結(jié)束后,開始事件循環(huán)(目前消息隊列中沒有任何東西,文件還沒讀完)
異步I/O讀取文件完畢,將消息壓入消息隊列(消息中含有文件內(nèi)容或者是出錯信息)
事件循環(huán)取得消息,執(zhí)行回調(diào)
程序退出。
這里借一張圖來說明nodejs的事件驅(qū)動模型(圖來自這里)
][2]
這里最后要說的一點就是如何手動將一個函數(shù)推入隊列,nodejs為我們提供了幾個比較方便的方法:
setTimeout()
process.nextTick()
setImmediate()
異步I/Onodejs中的異步I/O的操作是通過libuv這個庫來實現(xiàn)的,包含了window和linux下面的異步I/O實現(xiàn),博主也沒有研究過這個庫,感興趣的讀者可以移步到這里
問題答案好,到目前為止,已經(jīng)可以回答上面的問題了
question 1
為何timeout1和timeout2的結(jié)果會在end后面?
answer 1
因為此時timeout1和timeout2只是被異步函數(shù)推入到了隊列中,事件循環(huán)還是暫停狀態(tài)
question 2
為何timeout1和timeout2沒有輸出到終端?while(true)到底阻塞了什么?
answer 2
因為此處直接阻塞了事件循環(huán),還沒開始,就已經(jīng)被阻塞了
question 3,4
為什么timeout1中回調(diào)函數(shù)會阻塞timeout2中的回調(diào)函數(shù)的執(zhí)行?
為何timeout1的計算密集型工作將會阻塞timeout2的回調(diào)函數(shù)的執(zhí)行?
answer 3,4
因為該回調(diào)函數(shù)執(zhí)行返回事件循環(huán)才會繼續(xù)執(zhí)行,回調(diào)函數(shù)將會阻塞事件循環(huán)的運行
question 5
為何讀取文件的IO操作不會阻塞timeout2的執(zhí)行?
answer 5
因為IO操作是異步的,會開啟一個新的線程,不會阻塞到事件循環(huán)
參考文獻:
What exactly is a Node.js event loop tick?
What is the difference between a thread-based server and an event-based server?
Some confusion about nodejs threads
The JavaScript Event Loop: Explained
poll vs select vs event-based
Linux IO模式及 select、poll、epoll詳解
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://m.hztianpu.com/yun/79517.html
摘要:使用了一個事件驅(qū)動非阻塞式的模型,使其輕量又高效。的包管理器,是全球最大的開源庫生態(tài)系統(tǒng)。按照這個定義,之前所述的阻塞,非阻塞,多路復用信號驅(qū)動都屬于同步。 系列文章 Nodejs高性能原理(上) --- 異步非阻塞事件驅(qū)動模型Nodejs高性能原理(下) --- 事件循環(huán)詳解 前言 終于開始我nodejs的博客生涯了,先從基本的原理講起.以前寫過一篇瀏覽器執(zhí)行機制的文章,和nodej...
摘要:事件驅(qū)動在中,當某個執(zhí)行完畢后,會以事件的形式通知執(zhí)行操作的線程而線程去執(zhí)行對應事件的回調(diào)函數(shù)。為了處理異步,線程必須要有事件循環(huán),不斷的檢查有沒有事件要處理,并依次處理。其實在底層中,有一半的代碼,都是在處理事件隊列回調(diào)函數(shù)。 事件驅(qū)動 上一節(jié)中,我們提到異步I/O;當I/O處理完畢后,nodejs是怎樣知道I/O已經(jīng)完成了呢?又是怎樣去處理的呢?答案是:事件驅(qū)動(事件循環(huán))機制。 ...
原文 先說1.1總攬: Reactor模式 Reactor模式中的協(xié)調(diào)機制Event Loop Reactor模式中的事件分離器Event Demultiplexer 一些Event Demultiplexer處理不了的復雜I/O接口比如File I/O、DNS等 復雜I/O的解決方案 未完待續(xù) 前言 nodejs和其他編程平臺的區(qū)別在于如何去處理I/O接口,我們聽一個人介紹nodejs,總是...
摘要:瀏覽器拿到了簡書網(wǎng)的完整的頁面代碼,在解析和渲染這個頁面的時候,里面的圖片靜態(tài)資源,他們同樣也是一個個請求都需要經(jīng)過上面的主要的七個步驟。瀏覽器根據(jù)拿到的資源對頁面進行渲染,最終把一個完整的頁面呈現(xiàn)給了用戶。 瀏覽器訪問一個網(wǎng)站所經(jīng)歷的步驟 Chrome搜索自身的DNS緩存 搜索操作系統(tǒng)自身的DNS緩存(瀏覽器沒有找到緩存或緩存已經(jīng)失效)查看Chrome瀏覽器的DNS緩存信息(chr...
摘要:什么是在中什么時候需要是中的包管理器。允許我們?yōu)榘惭b各種模塊,這個包管理器為我們提供了安裝刪除等其它命令來管理模塊。 showImg(https://user-gold-cdn.xitu.io/2019/7/11/16bde5b2df52a924?w=4000&h=2667&f=jpeg&s=450648); 本文為您分享「Node.js 入門你需要知道的 10 個問題」這些問題可能也...
閱讀 2850·2021-09-24 09:47
閱讀 4466·2021-08-27 13:10
閱讀 3091·2019-08-30 15:44
閱讀 1358·2019-08-29 12:56
閱讀 2650·2019-08-28 18:07
閱讀 2701·2019-08-26 14:05
閱讀 2712·2019-08-26 13:41
閱讀 1328·2019-08-26 13:33