Node.js 是單進程單線程應用程序,但是通過事件和回調支持并發(fā)播急,所以性能非常高。
Node.js 的每一個 API 都是異步的喉镰,并作為一個獨立線程運行旅择,使用異步函數(shù)調用,并處理并發(fā)侣姆。
Node.js 基本上所有的事件機制都是用設計模式中觀察者模式實現(xiàn)。
Node.js 單線程類似進入一個while(true)的事件循環(huán)沉噩,直到?jīng)]有事件觀察者退出捺宗,每個異步事件都生成一個事件觀察者,如果有事件發(fā)生就調用該回調函數(shù).
事件驅動程序
Node.js 使用事件驅動模型川蒙,當web server接收到請求蚜厉,就把它關閉然后進行處理,然后去服務下一個web請求畜眨。
當這個請求完成昼牛,它被放回處理隊列,當?shù)竭_隊列開頭康聂,這個結果被返回給用戶贰健。
這個模型非常高效可擴展性非常強,因為webserver一直接受請求而不等待任何讀寫操作恬汁。(這也被稱之為非阻塞式IO或者事件驅動IO)
在事件驅動模型中伶椿,會生成一個主循環(huán)來監(jiān)聽事件,當檢測到事件時觸發(fā)回調函數(shù)氓侧。
整個事件驅動的流程就是這么實現(xiàn)的脊另,非常簡潔。有點類似于觀察者模式约巷,事件相當于一個主題(Subject)偎痛,而所有注冊到這個事件上的處理函數(shù)相當于觀察者(Observer)。
Node.js 有多個內置的事件独郎,我們可以通過引入 events 模塊踩麦,并通過實例化 EventEmitter 類來綁定和監(jiān)聽事件,如下實例:
// 引入 events 模塊
var events = require('events');
// 創(chuàng)建 eventEmitter 對象
var eventEmitter = new events.EventEmitter();
以下程序綁定事件處理程序:
// 綁定事件及事件的處理程序
eventEmitter.on('eventName', eventHandler);
我們可以通過程序觸發(fā)事件:
// 觸發(fā)事件
eventEmitter.emit('eventName');
實例
創(chuàng)建一個main.js文件囚聚,代碼如下所示:
// 引入 events 模塊
var events = require('events');
// 創(chuàng)建 eventEmitter 對象
var eventEmitter = new events.EventEmitter();
// 創(chuàng)建事件處理程序
var connectHandler = function connected() {// 連接處理器
console.log('連接成功靖榕。');
// 觸發(fā) data_recived 事件 (數(shù)據(jù)接收)
eventEmitter.emit('data_recived');
}
// 綁定connection 事件處理程序
eventEmitter.on('connection',connectHandler);
// 使用匿名函數(shù)綁定 data_recived 事件
eventEmitter.on('data_recived',function(){
console.log('數(shù)據(jù)接收成功。');
});
// 觸發(fā) connection 事件
eventEmitter.emit('connection');
console.log('程序執(zhí)行完畢顽铸。');
接下來讓我執(zhí)行以上代碼:
Node 應用程序時如何工作的茁计?
在 Node 應用程序中,執(zhí)行異步操作的函數(shù)將回調函數(shù)作為最后一個參數(shù), 回調函數(shù)接收錯誤對象作為第一個參數(shù)星压。
接下來讓我們來重新看下前面的實例践剂,創(chuàng)建一個 input.txt ,文件內容如下:
Twitch is learening Node.js .
創(chuàng)建 main.js 文件,代碼如下:
// 引入文件操作模塊
var fs = require('fs');
// 讀取文件內容
fs.readFile('input.txt',(err,data)=>{
if(err){
console.log(err.stack);
return;
}
console.log(data.toString());
});
console.log("程序執(zhí)行完畢娜膘!");
以上程序中 fs.readFile() 是異步函數(shù)用于讀取文件逊脯。 如果在讀取文件過程中發(fā)生錯誤,錯誤 err 對象就會輸出錯誤信息竣贪。
如果沒發(fā)生錯誤军洼,readFile 跳過 err 對象的輸出,文件內容就通過回調函數(shù)輸出演怎。
執(zhí)行以上代碼匕争,執(zhí)行結果如下:
D:\wamp\www\node_js\three>node fs.js
程序執(zhí)行完畢!
Twitch is learening Node.js .
接下來我們刪除 input.txt 文件爷耀,執(zhí)行結果如下所示:
D:\wamp\www\node_js\three>node fs.js
程序執(zhí)行完畢甘桑!
Error: ENOENT: no such file or directory, open 'D:\wamp\www\node_js\three\inputs.txt'
at Error (native)