詳解nodejs異步I/O和事件循環(huán)
事件驅動模型
現(xiàn)在我們來看看nodejs中的事件驅動和異步I/O是如何實現(xiàn)的.
nodejs是單線程(single thread)運行的,通過一個事件循環(huán)(event-loop)來循環(huán)取出消息隊列(event-queue)中的消息進行處理,處理過程基本上就是去調(diào)用該消息對應的回調(diào)函數(shù)。消息隊列就是當一個事件狀態(tài)發(fā)生變化時,就將一個消息壓入隊列中。
nodejs的時間驅動模型一般要注意下面幾個點:
- 因為是單線程的,所以當順序執(zhí)行js文件中的代碼的時候,事件循環(huán)是被暫停的。
- 當js文件執(zhí)行完以后,事件循環(huán)開始運行,并從消息隊列中取出消息,開始執(zhí)行回調(diào)函數(shù)
- 因為是單線程的,所以當回調(diào)函數(shù)被執(zhí)行的時候,事件循環(huán)是被暫停的
- 當涉及到I/O操作的時候,nodejs會開一個獨立的線程來進行異步I/O操作,操作結束以后將消息壓入消息隊列。
下面我們從一個簡單的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í)行結束后,開始事件循環(huán)(目前消息隊列中沒有任何東西,文件還沒讀完)
- 異步I/O讀取文件完畢,將消息壓入消息隊列(消息中含有文件內(nèi)容或者是出錯信息)
- 事件循環(huán)取得消息,執(zhí)行回調(diào)
- 程序退出。
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關文章
Node.js內(nèi)置模塊events事件監(jiān)聽發(fā)射詳解
這篇文章主要為大家介紹了Node.js內(nèi)置模塊events事件監(jiān)聽發(fā)射詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-02-02
Node.js數(shù)據(jù)流Stream之Readable流和Writable流用法
這篇文章介紹了Node.js數(shù)據(jù)流Stream之Readable流和Writable流的用法,文中通過示例代碼介紹的非常詳細。對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-07-07
nodejs進階(6)—連接MySQL數(shù)據(jù)庫示例
本篇文章主要介紹了nodejs進階(6)—連接MySQL數(shù)據(jù)庫示例,詳細的介紹了NodeJS操作MySQL數(shù)據(jù)庫,作為應用最為廣泛的開源數(shù)據(jù)庫則成為我們的首選,有興趣的可以了解一下。2017-01-01

