深入淺析Node.js 事件循環(huán)
Node.js 是單進(jìn)程單線程應(yīng)用程序,但是通過事件和回調(diào)支持并發(fā),所以性能非常高。
(來源于Javascript是單線程又是異步的,但是這種語言有個(gè)共同的特點(diǎn):它們是 event-driven 的。驅(qū)動(dòng)它們的 event 來自一個(gè)異構(gòu)的平臺(tái)。)
Node.js 的每一個(gè) API 都是異步的,并作為一個(gè)獨(dú)立線程運(yùn)行,使用異步函數(shù)調(diào)用,并處理并發(fā)。
Node.js 基本上所有的事件機(jī)制都是用設(shè)計(jì)模式中觀察者模式實(shí)現(xiàn)。
Node.js 單線程類似進(jìn)入一個(gè)while(true)的事件循環(huán),直到?jīng)]有事件觀察者退出,每個(gè)異步事件都生成一個(gè)事件觀察者,如果有事件發(fā)生就調(diào)用該回調(diào)函數(shù).
事件驅(qū)動(dòng)模型
Node.js 使用事件驅(qū)動(dòng)模型,當(dāng)webserver接收到請(qǐng)求,就把它關(guān)閉然后進(jìn)行處理,然后去服務(wù)下一個(gè)web請(qǐng)求。
當(dāng)這個(gè)請(qǐng)求完成,它被放回處理隊(duì)列,當(dāng)?shù)竭_(dá)隊(duì)列開頭,這個(gè)結(jié)果被返回給用戶。
這個(gè)模型非常高效可擴(kuò)展性非常強(qiáng),因?yàn)閣ebserver一直接受請(qǐng)求而不等待任何讀寫操作。
(這也被稱之為非阻塞式IO或者事件驅(qū)動(dòng)IO)
在事件驅(qū)動(dòng)模型中,會(huì)生成一個(gè)主循環(huán)來監(jiān)聽事件,當(dāng)檢測(cè)到事件時(shí)觸發(fā)回調(diào)函數(shù)。
整個(gè)事件驅(qū)動(dòng)的流程就是這么實(shí)現(xiàn)的,非常簡潔。有點(diǎn)類似于觀察者模式,事件相當(dāng)于一個(gè)主題(Subject),而所有注冊(cè)到這個(gè)事件上的處理函數(shù)相當(dāng)于觀察者(Observer)。
Node.js 有多個(gè)內(nèi)置的事件,我們可以通過引入 events 模塊,并通過實(shí)例化 EventEmitter 類來綁定和監(jiān)聽事件,如下實(shí)例:
//引入events模塊 var events = require('events'); //創(chuàng)建eventEmitter對(duì)象 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í)行完畢。");
以上代碼輸出的結(jié)果是:
連接成功。
數(shù)據(jù)接收成功。
程序執(zhí)行完畢。
上述的結(jié)果正如你想的一樣,通過這種事件驅(qū)動(dòng)模型我們就可以實(shí)現(xiàn)異步操作的。
(例如,我們可以一邊讀取文件,一邊執(zhí)行其他命令,在文件讀取完成后,我們將文件內(nèi)容作為回調(diào)函數(shù)的參數(shù)返回。這樣在執(zhí)行代碼時(shí)就沒有阻塞或等待文件 I/O 操作。這就大大提高了 Node.js 的性能,可以處理大量的并發(fā)請(qǐng)求。)
var fs = require("fs"); fs.readFile('input.txt', function (err, data) { if (err){ console.log(err.stack); return; } console.log(data.toString()); }); console.log("程序執(zhí)行完畢。");
以上代碼輸出的結(jié)果是:
程序執(zhí)行完畢。
input.txt的內(nèi)容。
從上述代碼可以看出fs.readFile可以分發(fā)事件,當(dāng)然Node.js里面的許多對(duì)象都會(huì)分發(fā)事件,一個(gè)net.Server對(duì)象會(huì)在每次有新連接時(shí)也會(huì)分發(fā)一個(gè)事件, 所有這些產(chǎn)生事件的對(duì)象都是 events.EventEmitter 的實(shí)例。
不過大多數(shù)時(shí)候我們不會(huì)直接使用 EventEmitter,而是在對(duì)象中繼承它。包括 fs、net、 http 在內(nèi)的,只要是支持事件響應(yīng)的核心模塊都是 EventEmitter 的子類。
為什么要這樣做呢?原因有兩點(diǎn):
首先,具有某個(gè)實(shí)體功能的對(duì)象實(shí)現(xiàn)事件符合語義, 事件的監(jiān)聽和發(fā)射應(yīng)該是一個(gè)對(duì)象的方法。
其次 JavaScript 的對(duì)象機(jī)制是基于原型的,支持 部分多重繼承,繼承 EventEmitter 不會(huì)打亂對(duì)象原有的繼承關(guān)系。
下面給大家分享我的第一個(gè)NodeJs項(xiàng)目
Node.js的安裝通常有兩種方式:自己編譯源代碼和使用編譯好的文件,我這里使用編譯好的文件
目前我的home目錄下有剛下載來的node-v4.2.3-linux-x64
1.首先解壓縮
tar xvf node-v4.2.3-linux-x64
2.設(shè)置鏈接,設(shè)置鏈接的目的在于任何路徑都能夠用到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文件夾下建個(gè)目錄叫mynodeproj,然后再建了個(gè)文件叫server.js(當(dāng)然也可以起別的名字)
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ā)送響應(yīng)數(shù)據(jù) "Hello World" response.end('Hello World\n'); }).listen(8888); //終端打印如下信息 console.log('Server running at http://127.0.0.1:8888/');
4.使用node命令執(zhí)行以上代碼
node server.js
執(zhí)行上述命令之后會(huì)在命令行中顯示"Server runnint at
因?yàn)槲疫@個(gè)服務(wù)器是在阿里云里跑起來的,所以只要在外面的瀏覽器中鍵入阿里云IP+端口就可以訪問了。
- nodeJs事件循環(huán)運(yùn)行代碼解析
- 帶你了解NodeJS事件循環(huán)
- Nodejs監(jiān)控事件循環(huán)異常示例詳解
- 詳解nodejs異步I/O和事件循環(huán)
- 我的Node.js學(xué)習(xí)之路(三)--node.js作用、回調(diào)、同步和異步代碼 以及事件循環(huán)
- Node.js事件循環(huán)(Event Loop)和線程池詳解
- 深入理解Node.js 事件循環(huán)和回調(diào)函數(shù)
- 小結(jié)Node.js中非阻塞IO和事件循環(huán)
- 實(shí)例分析JS與Node.js中的事件循環(huán)
- nodejs?快速入門之事件循環(huán)
相關(guān)文章
npm?install總是卡住不動(dòng)問題的解決辦法
在我們安裝完Node.js之后,需要使用npm命令來安裝一些工具,下面這篇文章主要給大家介紹了關(guān)于npm?install總是卡住不動(dòng)問題的解決辦法,需要的朋友可以參考下2022-05-05nodejs入門教程五:連接數(shù)據(jù)庫的方法分析
這篇文章主要介紹了nodejs入門教程之連接數(shù)據(jù)庫的方法,結(jié)合實(shí)例形式分析了nodejs連接數(shù)據(jù)庫的具體步驟與相關(guān)操作技巧,需要的朋友可以參考下2017-04-04NodeJs Express框架實(shí)現(xiàn)服務(wù)器接口詳解
最近學(xué)習(xí)了基于前后端分離的開發(fā)模式,我前端使用Vue框架,后端使用nodejs開發(fā)API接口,下面這篇文章主要給大家介紹了關(guān)于nodejs使用Express框架寫后端接口的相關(guān)資料,需要的朋友可以參考下2022-08-08node.js將MongoDB數(shù)據(jù)同步到MySQL的步驟
這篇文章主要給大家介紹了關(guān)于node.js將MongoDB數(shù)據(jù)同步到MySQL的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧。2017-12-12Node.js如何自動(dòng)審核團(tuán)隊(duì)的代碼
在項(xiàng)目開發(fā)中,統(tǒng)一團(tuán)隊(duì)的代碼風(fēng)格很重要,本文介紹如何用Node.js來自動(dòng)審核,來提高您的開發(fā)速度。2016-07-07