簡單的說 Node.js 就是運行在服務端的 JavaScript。node
Node.js 是一個基於Chrome JavaScript 運行時創建的一個平臺。web
Node.js是一個事件驅動I/O服務端JavaScript環境,基於Google的V8引擎,V8引擎執行Javascript的速度很是快,性能很是好。編程
Node.js 是單進程單線程應用程序,可是經過事件和回調支持併發,因此性能很是高。設計模式
Node.js 的每個 API 都是異步的,並做爲一個獨立線程運行,使用異步函數調用,並處理併發。併發
Node.js 基本上全部的事件機制都是用設計模式中觀察者模式實現。異步
Node.js 單線程相似進入一個while(true)的事件循環,直到沒有事件觀察者退出,每一個異步事件都生成一個事件觀察者,若是有事件發生就調用該回調函數.異步編程
Node.js 使用事件驅動模型,當web server接收到請求,就把它關閉而後進行處理,而後去服務下一個web請求。函數
當這個請求完成,它被放回處理隊列,當到達隊列開頭,這個結果被返回給用戶。性能
這個模型很是高效可擴展性很是強,由於webserver一直接受請求而不等待任何讀寫操做。(這也被稱之爲非阻塞式IO或者事件驅動IO)ui
在事件驅動模型中,會生成一個主循環來監聽事件,當檢測到事件時觸發回調函數。
整個事件驅動的流程就是這麼實現的,很是簡潔。有點相似於觀察者模式,事件至關於一個主題(Subject),而全部註冊到這個事件上的處理函數至關於觀察者(Observer)。
Node.js 有多個內置的事件,咱們能夠經過引入 events 模塊,並經過實例化 EventEmitter 類來綁定和監聽事件,以下實例:
// 引入 events 模塊var events =require('events');// 建立 eventEmitter 對象var eventEmitter =new events.EventEmitter();
如下程序綁定事件處理程序:
// 綁定事件及事件的處理程序 eventEmitter.on('eventName', eventHandler);
咱們能夠經過程序觸發事件:
// 觸發事件 eventEmitter.emit('eventName');
建立 main.js 文件,代碼以下所示:
// 引入 events 模塊var events =require('events');// 建立 eventEmitter 對象var eventEmitter =new events.EventEmitter();// 建立事件處理程序var connectHandler =function connected(){ console.log('鏈接成功。');// 觸發 data_received 事件 eventEmitter.emit('data_received');}// 綁定 connection 事件處理程序 eventEmitter.on('connection', connectHandler);// 使用匿名函數綁定 data_received 事件 eventEmitter.on('data_received',function(){ console.log('數據接收成功。');});// 觸發 connection 事件 eventEmitter.emit('connection'); console.log("程序執行完畢。");
接下來讓咱們執行以上代碼:
$ node main.js 鏈接成功。數據接收成功。程序執行完畢。
在 Node 應用程序中,執行異步操做的函數將回調函數做爲最後一個參數, 回調函數接收錯誤對象做爲第一個參數。
Node.js 異步編程的直接體現就是回調。
異步編程依託於回調來實現,但不能說使用了回調後程序就異步化了。
回調函數在完成任務後就會被調用,Node 使用了大量的回調函數,Node 全部 API 都支持回調函數。
例如,咱們能夠一邊讀取文件,一邊執行其餘命令,在文件讀取完成後,咱們將文件內容做爲回調函數的參數返回。這樣在執行代碼時就沒有阻塞或等待文件 I/O 操做。這就大大提升了 Node.js 的性能,能夠處理大量的併發請求。
建立一個文件 input.txt ,內容以下:
菜鳥教程官網地址:www.runoob.com
建立 main.js 文件, 代碼以下:
var fs =require("fs");var data = fs.readFileSync('input.txt'); console.log(data.toString()); console.log("程序執行結束!");
以上代碼執行結果以下:
$ node main.js 菜鳥教程官網地址:www.runoob.com 程序執行結束!
建立一個文件 input.txt ,內容以下:
菜鳥教程官網地址:www.runoob.com
建立 main.js 文件, 代碼以下:
var fs =require("fs"); fs.readFile('input.txt',function(err, data){if(err)return console.error(err); console.log(data.toString());}); console.log("程序執行結束!");
以上代碼執行結果以下:
$ node main.js 程序執行結束!菜鳥教程官網地址:www.runoob.com
以上兩個實例咱們瞭解了阻塞與非阻塞調用的不一樣。第一個實例在文件讀取完後才執行完程序。 第二個實例咱們不須要等待文件讀取完,這樣就能夠在讀取文件時同時執行接下來的代碼,大大提升了程序的性能。
所以,阻塞是按順序執行的,而非阻塞是不須要按順序的,因此若是須要處理回調函數的參數,咱們就須要寫在回調函數內。