首頁 >web前端 >js教程 >深入淺析Node.js 事件循環_node.js

深入淺析Node.js 事件循環_node.js

WBOY
WBOY原創
2016-05-16 15:24:161525瀏覽

Node.js 是單進程單線程應用程序,但透過事件和回調支援並發,所以效能非常高。

(來自Javascript是單線程又是異步的,但是這種語言有個共同的特點:它們是 event-driven 的。驅動它們的 event 來自異構的平台。)

Node.js 的每一個 API 都是非同步的,並作為一個獨立執行緒運行,使用非同步函數調用,並處理並發。

Node.js 基本上所有的事件機制都是用設計模式中觀察者模式實作。

Node.js 單執行緒類似進入一個while(true)的事件循環,直到沒有事件觀察者退出,每個非同步事件都會產生一個事件觀察者,如果有事件發生就呼叫該回調函數.

事件驅動模型

Node.js 使用事件驅動模型,當webserver接收到請求,就把它關閉然後進行處理,然後去服務下一個web請求。

當這個請求完成,它被放回處理佇列,當到達佇列開頭,這個結果被回傳給使用者。

這個模型非常有效率可擴充性非常強,因為webserver一直接受請求而不等待任何讀寫操作。
(這也稱為非阻塞式IO或事件驅動IO)

在事件驅動模型中,會產生一個主循環來監聽事件,當偵測到事件時觸發回呼函數。

整個事件驅動的流程就是這麼實現的,非常簡潔。有點類似觀察者模式,事件相當於一個主題(Subject),而所有註冊到這個事件上的處理函數相當於觀察者(Observer)。

Node.js 有多個內建的事件,我們可以透過引入 events 模組,並透過實例化 EventEmitter 類別來綁定和監聽事件,如下實例:

//引入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("程序执行完毕。"); 

以上程式碼輸出的結果是:

連線成功。

資料接收成功。

程式執行完畢。

上述的結果正如你想的一樣,透過這種事件驅動模型我們就可以實現非同步操作的。

(例如,我們可以一邊讀取文件,一邊執行其他命令,在文件讀取完成後,我們將文件內容作為回調函數的參數傳回。這樣在執行程式碼時就沒有阻塞或等待文件I/O操作。

var fs = require("fs");
fs.readFile('input.txt', function (err, data) {
 if (err){
 console.log(err.stack);
 return;
 }
 console.log(data.toString());
});
console.log("程序执行完毕。"); 

以上程式碼輸出的結果是:

程式執行完畢。

input.txt的內容。

從上述程式碼可以看出fs.readFile可以分發事件,當然Node.js裡面的許多物件都會分發事件,一個net.Server物件會在每次有新連接時也會分發一個事件, 所有這些產生事件的物件都是events.EventEmitter 的實例。


不過大多數時候我們不會直接使用 EventEmitter,而是在物件中繼承它。包括 fs、net、 http 在內的,只要是支援事件回應的核心模組都是 EventEmitter 的子類別。


為什麼要這樣做呢?原因有兩點:

首先,具有某個實體功能的物件實作事件符合語義, 事件的監聽和發射應該是一個物件的方法。

其次 JavaScript 的物件機制是基於原型的,支援 部分多重繼承,繼承 EventEmitter 不會打亂物件原有的繼承關係。

下面要跟大家分享我的第一個NodeJs專案

Node.js的安裝通常有兩種方式:自己編譯原始碼和使用編譯好的文件,我這裡使用編譯好的文件

目前我的home目錄下有剛下載來的node-v4.2.3-linux-x64

1.先解壓縮

複製程式碼 程式碼如下:
tar xvf node-v4.2.3-linux-x64

2.設定鏈接,設定連結的目的在於任何路徑都能夠用到node指令

複製程式碼 程式碼如下:
ln -s /home/node-v4.2.3-linux-x64/bin/node /usr/local/bin/node
ln -s /home/node-v4.2.3-linux-x64/bin/npm /usr/local/bin/npm

3.在home資料夾下建目錄叫mynodeproj,然後再建了個檔案叫server.js(當然也可以起別的名字)

var http = require('http');
http.createServer(function (request, response) {
 //发送 HTTP 头部 
 //HTTP 状态值: 200 : OK
 //内容类型: text/plain
 response.writeHead(200, {'Content-Type': 'text/plain'});
 //发送响应数据 "Hello World"
 response.end('Hello World\n');
}).listen(8888);
//终端打印如下信息
console.log('Server running at http://127.0.0.1:8888/'); 

4.使用node指令執行以上程式碼

node server.js

執行上述指令之後會在指令列中顯示"Server runnint at http://127.0.0.1:8888/",這樣表示程式已經執行成功了!

因為我這個伺服器是在阿里雲裡跑起來的,所以只要在外面的瀏覽器中鍵入阿里雲IP+端口就可以訪問了。

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn