1、初級搭建
1)引入 required 模塊
使用 require 指令來載入 http 模塊哪自,并將實例化的 HTTP 賦值給變量 http
var http = require("http");
2)創(chuàng)建服務器
使用 http.createServer() 方法創(chuàng)建服務器务漩,并使用 listen 方法綁定 8888 端口内斯。 函數(shù)通過 request, response 參數(shù)來接收和響應數(shù)據(jù)蒲每。
創(chuàng)建一個叫 server.js 的文件,并寫入以下代碼:
var http = require('http');
http.createServer(function (request, response) {
// 發(fā)送 HTTP 頭部
// HTTP 狀態(tài)值: 200 : OK
// 內(nèi)容類型: text/plain
response.writeHead(200, {'Content-Type': 'text/plain'});
// 發(fā)送響應數(shù)據(jù) "Hello World"
response.end('Hello World\n');
}).listen(8888);
// 終端打印如下信息
console.log('Server running at http://127.0.0.1:8888/');
2扫夜、Node.js 回調(diào)函數(shù)
Node.js 異步編程的直接體現(xiàn)就是回調(diào)。
異步編程依托于回調(diào)來實現(xiàn)驰徊,回調(diào)函數(shù)在完成任務后就會被調(diào)用笤闯,Node 使用了大量的回調(diào)函數(shù),Node 所有 API 都支持回調(diào)函數(shù)棍厂。
例如颗味,我們可以一邊讀取文件,一邊執(zhí)行其他命令牺弹,在文件讀取完成后浦马,我們將文件內(nèi)容作為回調(diào)函數(shù)的參數(shù)返回。這樣在執(zhí)行代碼時就沒有阻塞或等待文件 I/O 操作张漂。這就大大提高了 Node.js 的性能晶默,可以處理大量的并發(fā)請求。
阻塞代碼實例
創(chuàng)建一個文件 input.txt 航攒,內(nèi)容如下:
my name is 123
創(chuàng)建 main.js 文件, 代碼如下:
var fs = require("fs");
var data = fs.readFileSync('input.txt');
console.log(data.toString());
console.log("程序執(zhí)行結(jié)束!");
以上代碼執(zhí)行結(jié)果如下:
$ node main.js
my name is 123\
程序執(zhí)行結(jié)束!
非阻塞代碼實例
創(chuàng)建一個文件 input.txt 磺陡,內(nèi)容如下:
my name is 123
創(chuàng)建 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("程序執(zhí)行結(jié)束!");
以上代碼執(zhí)行結(jié)果如下:
$ node main.js
程序執(zhí)行結(jié)束!
my name is 123
阻塞與非阻塞調(diào)用的不同。第一個實例在文件讀取完后才執(zhí)行完程序。 第二個實例我們不需要等待文件讀取完仅政,這樣就可以在讀取文件時同時執(zhí)行接下來的代碼垢油,大大提高了程序的性能。
因此圆丹,阻塞是按順序執(zhí)行的滩愁,而非阻塞是不需要按順序的,所以如果需要處理回調(diào)函數(shù)的參數(shù)辫封,我們就需要寫在回調(diào)函數(shù)內(nèi)硝枉。
3、Node.js 事件循環(huán)
// 引入 events 模塊
var events = require('events');
// 創(chuàng)建 eventEmitter 對象
var eventEmitter = new events.EventEmitter();
// 創(chuàng)建事件處理程序
var connectHandler = function connected() {
console.log('連接成功倦微。');
// 觸發(fā) data_received 事件
eventEmitter.emit('data_received');
}
// 綁定 connection 事件處理程序
eventEmitter.on('connection', connectHandler);
// 使用匿名函數(shù)綁定 data_received 事件
eventEmitter.on('data_received', function(){
console.log('數(shù)據(jù)接收成功妻味。');
});
// 觸發(fā) connection 事件
eventEmitter.emit('connection');
console.log("程序執(zhí)行完畢。");
Node.js 是單進程單線程應用程序欣福,但是通過事件和回調(diào)支持并發(fā)责球,所以性能非常高。
Node.js 的每一個 API 都是異步的拓劝,并作為一個獨立線程運行雏逾,使用異步函數(shù)調(diào)用,并處理并發(fā)郑临。
Node.js 基本上所有的事件機制都是用設計模式中觀察者模式實現(xiàn)栖博。
Node.js 單線程類似進入一個while(true)的事件循環(huán),直到?jīng)]有事件觀察者退出厢洞,每個異步事件都生成一個事件觀察者仇让,如果有事件發(fā)生就調(diào)用該回調(diào)函數(shù).
事件驅(qū)動程序
Node.js 使用事件驅(qū)動模型,當web server接收到請求躺翻,就把它關閉然后進行處理丧叽,然后去服務下一個web請求。
當這個請求完成获枝,它被放回處理隊列蠢正,當?shù)竭_隊列開頭,這個結(jié)果被返回給用戶省店。
這個模型非常高效可擴展性非常強嚣崭,因為webserver一直接受請求而不等待任何讀寫操作。(這也被稱之為非阻塞式IO或者事件驅(qū)動IO)
在事件驅(qū)動模型中懦傍,會生成一個主循環(huán)來監(jiān)聽事件雹舀,當檢測到事件時觸發(fā)回調(diào)函數(shù)。