溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

使用node實(shí)現(xiàn)事件循環(huán)

發(fā)布時(shí)間:2020-11-02 16:09:29 來源:億速云 閱讀:156 作者:Leah 欄目:開發(fā)技術(shù)

使用node實(shí)現(xiàn)事件循環(huán)?很多新手對(duì)此不是很清楚,為了幫助大家解決這個(gè)難題,下面小編將為大家詳細(xì)講解,有這方面需求的人可以來學(xué)習(xí)下,希望你能有所收獲。

事件循環(huán)

Node.js 是單進(jìn)程單線程應(yīng)用程序,但是因?yàn)?V8 引擎提供的異步執(zhí)行回調(diào)接口,通過這些接口可以處理大量的并發(fā),所以性能非常高。

Node.js 幾乎每一個(gè) API 都是支持回調(diào)函數(shù)的。

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ù).

進(jìn)程:CPU執(zhí)行任務(wù)的模塊

線程:模塊中的最小單元

舉個(gè)通俗的例子:

cpu比作我們每個(gè)人,到飯點(diǎn)吃飯了??梢渣c(diǎn)很多菜(cpu中的進(jìn)程):宮保雞丁,魚香肉絲,酸辣土豆絲。每樣菜具體包含了哪些內(nèi)容(cpu每個(gè)進(jìn)程中的線程):宮保雞丁(詳情:黃瓜、胡蘿卜、雞肉、花生米)。而詳情構(gòu)成了宮保雞丁這道菜,吃了以后不餓。就可以干活了,cpu中的進(jìn)程里的線程也是同理。當(dāng)線程完成自己的內(nèi)容將結(jié)果返回給進(jìn)程,進(jìn)程返回給cpu的時(shí)候。cpu就能處理日常需求。

  • 單進(jìn)程單線程:一盤炒苦瓜,里面只有苦瓜。
  • 單進(jìn)程多線程:一盤宮保雞丁,里面有黃瓜、胡蘿卜、雞肉、花生米

事件驅(qū)動(dòng)程序

Node.js使用事件驅(qū)動(dòng)模型,當(dāng)web server接收到請(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)?webserver 一直接受請(qǐng)求而不等待任何讀寫操作。(這也稱之為非阻塞式IO或者事件驅(qū)動(dòng)IO)

在事件驅(qū)動(dòng)模型中,會(huì)生成一個(gè)主循環(huán)來監(jiān)聽事件,當(dāng)檢測(cè)到事件時(shí)觸發(fā)回調(diào)函數(shù)。

使用node實(shí)現(xiàn)事件循環(huán)

整個(gè)事件驅(qū)動(dòng)的流程就是這么實(shí)現(xiàn)的,非常簡(jiǎ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模塊
let events = require('events');
//創(chuàng)建eventEmitter對(duì)象
let eventEmitter = new events.EventEmitter();

//綁定事件以及事件處理程序
eventEmitter.on('eventName',eventHandler);

//通過程序觸發(fā)事件
eventEmitter.emit('eventName')

示例:

//引入events模塊
let events = require('events');

//創(chuàng)建eventEmitter對(duì)象
let eventEmitter = new events.EventEmitter();

//創(chuàng)建事件處理程序
let connectHandle = function connected() {
 console.log('連接成功');
 //觸發(fā)data_received事件
 eventEmitter.emit('data_received')
}

//綁定connection事件處理程序
eventEmitter.on('connection', connectHandle);

//使用匿名函數(shù)綁定data_received事件
eventEmitter.on('data_received', function () {
 console.log('數(shù)據(jù)接收成功');
})

//觸發(fā)connecttion事件
eventEmitter.emit('connection');
console.log('程序執(zhí)行完畢');


// 執(zhí)行結(jié)果:
// 連接成功
// 數(shù)據(jù)接收成功
// 程序執(zhí)行完畢

eventEmitter.emit 是觸發(fā)事件(事件請(qǐng)求),eventEmitter.on是綁定處理事件的處理器(事件處理),事件的請(qǐng)求和處理是分開的,所以是異步。

EventEmitter

node.js所有的異步I/O操作在完成時(shí)都會(huì)發(fā)送一個(gè)事件到事件隊(duì)列

node.js里面的許多對(duì)象都會(huì)分發(fā)事件:一個(gè)net.Server對(duì)象會(huì)在每次有新連接時(shí)觸發(fā)的一個(gè)事件,一個(gè)fs.readStream對(duì)象會(huì)在文件被打開的時(shí)候觸發(fā)一個(gè)事件。所有這些產(chǎn)生事件的對(duì)象都是events.EventEmitter的實(shí)例

events 模塊只提供了一個(gè)對(duì)象: events.EventEmitter。EventEmitter 的核心就是事件觸發(fā)與事件監(jiān)聽器功能的封裝。

EventEmitter 對(duì)象如果在實(shí)例化時(shí)發(fā)生錯(cuò)誤,會(huì)觸發(fā) error 事件。當(dāng)添加新的監(jiān)聽器時(shí),newListener 事件會(huì)觸發(fā),當(dāng)監(jiān)聽器被移除時(shí),removeListener 事件被觸發(fā)。

簡(jiǎn)單用法

var EventEmitter = require('events').EventEmitter; 
var event = new EventEmitter(); 
event.on('some_event', function() { 
  console.log('some_event 事件觸發(fā)'); 
}); 
setTimeout(function() { 
  event.emit('some_event'); 
}, 1000); 

運(yùn)行這段代碼,1 秒后控制臺(tái)輸出了 ‘some_event 事件觸發(fā)'。
其原理是 event 對(duì)象注冊(cè)了事件 some_event 的一個(gè)監(jiān)聽器,
然后我們通過 setTimeout 在 1000 毫秒以后向 event 對(duì)象發(fā)送事件 some_event,此時(shí)會(huì)調(diào)用some_event 的監(jiān)聽器。

EventEmitter 的每個(gè)事件由一個(gè)事件名和若干個(gè)參數(shù)組成,事件名是一個(gè)字符串,通常表達(dá)一定的語義。對(duì)于每個(gè)事件,EventEmitter 支持 若干個(gè)事件監(jiān)聽器。

當(dāng)事件觸發(fā)時(shí),注冊(cè)到這個(gè)事件的事件監(jiān)聽器被依次調(diào)用,事件參數(shù)作為回調(diào)函數(shù)參數(shù)傳遞。

var events = require('events'); 
var emitter = new events.EventEmitter(); 
emitter.on('someEvent', function(arg1, arg2) { 
  console.log('listener1', arg1, arg2); 
}); 
emitter.on('someEvent', function(arg1, arg2) { 
  console.log('listener2', arg1, arg2); 
}); 
emitter.emit('someEvent', 'arg1 參數(shù)', 'arg2 參數(shù)'); 
//輸出:
// listener1 arg1 參數(shù) arg2 參數(shù)
// listener2 arg1 參數(shù) arg2 參數(shù)

以上例子中,emitter 為事件 someEvent 注冊(cè)了兩個(gè)事件監(jiān)聽器,然后觸發(fā)了 someEvent 事件。

運(yùn)行結(jié)果中可以看到兩個(gè)事件監(jiān)聽器回調(diào)函數(shù)被先后調(diào)用。 這就是EventEmitter最簡(jiǎn)單的用法。

EventEmitter 提供了多個(gè)屬性,如 on 和 emit。on 函數(shù)用于綁定事件函數(shù),emit 屬性用于觸發(fā)一個(gè)事件。、

EventEmitter屬性

使用node實(shí)現(xiàn)事件循環(huán)

測(cè)試

/*
 * @Author: angula
 * @Date: 2020-09-21 22:29:18
 * @LastEditTime: 2020-09-22 11:27:56
 * @FilePath: \JS\Github-前端知識(shí)總結(jié)倉(cāng)庫(kù)\studySummary\Node.js學(xué)習(xí)筆記\事件循環(huán)\index2.js
 */
let events = require('events');
let eventEmitter = new events.EventEmitter();

// 監(jiān)聽器1
let listener1 = function listener1() {
 console.log('監(jiān)聽器listener1啟動(dòng)。。。');
}

// 監(jiān)聽器2
let listener2 = function listener2() {
 console.log('監(jiān)聽器listener2啟動(dòng)。。。');
}

// 綁定connection事件,處理函數(shù)為listener1
eventEmitter.addListener('connection', listener1);
// 綁定connection事件,處理函數(shù)為listener2
eventEmitter.on('connection', listener2);


//類,返回監(jiān)聽器的數(shù)量

let eventListeners = eventEmitter.listenerCount('connection');
console.log(eventListeners + '個(gè)監(jiān)聽器監(jiān)聽連接事件。');


//處理connection事件
eventEmitter.emit('connection');

// 移除綁定的listener1
eventEmitter.removeListener('connection', listener1);
console.log('listener1不再受監(jiān)聽');

//觸發(fā)連接事件
eventEmitter.emit('connection');

eventListeners = eventEmitter.listenerCount('connection');
console.log(eventListeners + '個(gè)監(jiān)聽器連接事件');

console.log('程序執(zhí)行完畢');

執(zhí)行結(jié)果:

使用node實(shí)現(xiàn)事件循環(huán)

看完上述內(nèi)容是否對(duì)您有幫助呢?如果還想對(duì)相關(guān)知識(shí)有進(jìn)一步的了解或閱讀更多相關(guān)文章,請(qǐng)關(guān)注億速云行業(yè)資訊頻道,感謝您對(duì)億速云的支持。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI