Node.js<二>

Node.js 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í)例。

EventEmitter 類

events 模塊只提供了一個(gè)對(duì)象: events.EventEmitter。EventEmitter 的核心就是事件觸發(fā)與事件監(jiān)聽器功能的封裝。
你可以通過(guò)require("events");來(lái)訪問(wèn)該模塊。

// 引入 events 模塊
var events = require('events');
// 創(chuàng)建 eventEmitter 對(duì)象
var eventEmitter = new events.EventEmitter();

EventEmitter 對(duì)象如果在實(shí)例化時(shí)發(fā)生錯(cuò)誤,會(huì)觸發(fā) error 事件。當(dāng)添加新的監(jiān)聽器時(shí),newListener 事件會(huì)觸發(fā),當(dāng)監(jiān)聽器被移除時(shí),removeListener 事件被觸發(fā)。
下面我們用一個(gè)簡(jiǎn)單的例子說(shuō)明 EventEmitter 的用法:

//event.js 文件
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); 

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



EventEmitter 的每個(gè)事件由一個(gè)事件名和若干個(gè)參數(shù)組成,事件名是一個(gè)字符串,通常表達(dá)一定的語(yǔ)義。對(duì)于每個(gè)事件,EventEmitter 支持 若干個(gè)事件監(jiān)聽器。
當(dāng)事件觸發(fā)時(shí),注冊(cè)到這個(gè)事件的事件監(jiān)聽器被依次調(diào)用,事件參數(shù)作為回調(diào)函數(shù)參數(shù)傳遞。
讓我們以下面的例子解釋這個(gè)過(guò)程:

//event.js 文件
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ù)'); 

執(zhí)行以上代碼,運(yùn)行的結(jié)果如下:



以上例子中,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è)事件。接下來(lái)我們來(lái)具體看下 EventEmitter 的屬性介紹。

方法

<li>addListener(event, listener) 為指定事件添加一個(gè)監(jiān)聽器到監(jiān)聽器數(shù)組的尾部。
<li>on(event, listener)為指定事件注冊(cè)一個(gè)監(jiān)聽器,接受一個(gè)字符串 event 和一個(gè)回調(diào)函數(shù)。
<li>once(event, listener)為指定事件注冊(cè)一個(gè)單次監(jiān)聽器,即 監(jiān)聽器最多只會(huì)觸發(fā)一次,觸發(fā)后立刻解除該監(jiān)聽器。
<li>removeListener(event, listener)移除指定事件的某個(gè)監(jiān)聽器,監(jiān)聽器必須是該事件已經(jīng)注冊(cè)過(guò)的監(jiān)聽器。它接受兩個(gè)參數(shù),第一個(gè)是事件名稱,第二個(gè)是回調(diào)函數(shù)名稱。
<li>removeAllListeners([event])移除所有事件的所有監(jiān)聽器, 如果指定事件,則移除指定事件的所有監(jiān)聽器。
<li>setMaxListeners(n)默認(rèn)情況下, EventEmitters 如果你添加的監(jiān)聽器超過(guò) 10 個(gè)就會(huì)輸出警告信息。 setMaxListeners 函數(shù)用于提高監(jiān)聽器的默認(rèn)限制的數(shù)量。
<li>listeners(event)返回指定事件的監(jiān)聽器數(shù)組。
<li>emit(event, [arg1], [arg2], [...])按參數(shù)的順序執(zhí)行每個(gè)監(jiān)聽器,如果事件有注冊(cè)監(jiān)聽返回 true,否則返回 false。

類方法

listenerCount(emitter, event)返回指定事件的監(jiān)聽器數(shù)量。

事件

<li>newListener(event,listener) 該事件在添加新監(jiān)聽器時(shí)被觸發(fā)。
<li>removeListener(event,listener)從指定監(jiān)聽器數(shù)組中刪除一個(gè)監(jiān)聽器。需要注意的是,此操作將會(huì)改變處于被刪監(jiān)聽器之后的那些監(jiān)聽器的索引。

實(shí)例

以下實(shí)例通過(guò) connection(連接)事件演示了 EventEmitter 類的應(yīng)用。
創(chuàng)建 main.js 文件,代碼如下:

var events = require('events');
var eventEmitter = new events.EventEmitter();
// 監(jiān)聽器 #1
var listener1 = function listener1() {
   console.log('監(jiān)聽器 listener1 執(zhí)行。');
}
// 監(jiān)聽器 #2
var listener2 = function listener2() {
  console.log('監(jiān)聽器 listener2 執(zhí)行。');
}
// 綁定 connection 事件,處理函數(shù)為 listener1 
eventEmitter.addListener('connection', listener1);
// 綁定 connection 事件,處理函數(shù)為 listener2
eventEmitter.on('connection', listener2);
var eventListeners = require('events').EventEmitter.listenerCount(eventEmitter,'connection');
console.log(eventListeners + " 個(gè)監(jiān)聽器監(jiān)聽連接事件。");
// 處理 connection 事件 
eventEmitter.emit('connection');
// 移除監(jiān)綁定的 listener1 函數(shù)
eventEmitter.removeListener('connection', listener1);
console.log("listener1 不再受監(jiān)聽。");
// 觸發(fā)連接事件
eventEmitter.emit('connection');
eventListeners = require('events').EventEmitter.listenerCount(eventEmitter,'connection');
console.log(eventListeners + " 個(gè)監(jiān)聽器監(jiān)聽連接事件。");
console.log("程序執(zhí)行完畢。");

以上代碼,執(zhí)行結(jié)果如下所示


error 事件

EventEmitter 定義了一個(gè)特殊的事件 error,它包含了錯(cuò)誤的語(yǔ)義,我們?cè)谟龅?異常的時(shí)候通常會(huì)觸發(fā) error 事件。
當(dāng) error 被觸發(fā)時(shí),EventEmitter 規(guī)定如果沒有響 應(yīng)的監(jiān)聽器,Node.js 會(huì)把它當(dāng)作異常,退出程序并輸出錯(cuò)誤信息。
我們一般要為會(huì)觸發(fā) error 事件的對(duì)象設(shè)置監(jiān)聽器,避免遇到錯(cuò)誤后整個(gè)程序崩潰。例如:

var events = require('events'); 
var emitter = new events.EventEmitter(); 
emitter.emit('error'); 

運(yùn)行時(shí)會(huì)顯示以下錯(cuò)誤:


繼承 EventEmitter

大多數(shù)時(shí)候我們不會(huì)直接使用 EventEmitter,而是在對(duì)象中繼承它。包括 fs、net、 http 在內(nèi)的,只要是支持事件響應(yīng)的核心模塊都是 EventEmitter 的子類。
為什么要這樣做呢?原因有兩點(diǎn):
首先,具有某個(gè)實(shí)體功能的對(duì)象實(shí)現(xiàn)事件符合語(yǔ)義, 事件的監(jiān)聽和發(fā)射應(yīng)該是一個(gè)對(duì)象的方法。
其次 JavaScript 的對(duì)象機(jī)制是基于原型的,支持 部分多重繼承,繼承 EventEmitter 不會(huì)打亂對(duì)象原有的繼承關(guān)系。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • https://nodejs.org/api/documentation.html 工具模塊 Assert 測(cè)試 ...
    KeKeMars閱讀 6,597評(píng)論 0 6
  • Module definition patterns 除了作為加載依賴的機(jī)制之外,模塊系統(tǒng)也是一種用于定義AP...
    宮若石閱讀 556評(píng)論 0 0
  • 內(nèi)容來(lái)自《Node.js開發(fā)指南》 核心模塊是 Node.js 的心臟,它由一些精簡(jiǎn)而高效的庫(kù)組成,為 Node....
    angelwgh閱讀 965評(píng)論 0 1
  • “齊步走!” 噔噔噔噔噔噔 …… 我僵硬的站著,看著對(duì)面走過(guò)來(lái)的同學(xué)還沒有停下來(lái)的感覺,大眼瞪小眼,忽...
    everstory閱讀 216評(píng)論 0 3
  • 文/小美 我生長(zhǎng)在一個(gè)農(nóng)村里,父親是一個(gè)極其癡迷風(fēng)水,每頓都要喝一兩白酒,整天無(wú)所事事的人,母親是一個(gè)老實(shí)膽小的,...
    小美2016閱讀 1,450評(píng)論 12 2

友情鏈接更多精彩內(nèi)容