Node文檔筆記-events(事件)

1、events(事件)

對(duì)于大多數(shù)的Node.js核心API采用的是異步事件驅(qū)動(dòng)的架構(gòu),其中某些類型的對(duì)象(觸發(fā)器)會(huì)周期性的觸發(fā)命名事件來(lái)調(diào)用函數(shù)對(duì)象(監(jiān)聽器),下面看一個(gè)例子:

//引入events事件模塊,myEmitter對(duì)象繼承EventEmitter類
const EventEmitter = require('events');
class MyEmitter extends EventEmitter {}

//eventEmitter.on() 方法用于注冊(cè)監(jiān)聽器,eventEmitter.emit() 方法用于觸發(fā)事件,這兩個(gè)事件一般配合使用
const myEmitter = new MyEmitter();
myEmitter.on('event', () => {
  console.log('觸發(fā)了一個(gè)事件!');
});
myEmitter.emit('event');

2、監(jiān)聽器傳入?yún)?shù)與this

eventEmitter.emit()方法允許將任意參數(shù)傳給監(jiān)聽器函數(shù)eventEmitter.on().當(dāng)一個(gè)普通的監(jiān)聽函數(shù)被EventEmitter調(diào)用時(shí),標(biāo)準(zhǔn)的this關(guān)鍵詞將被設(shè)置指定監(jiān)聽器所附加的 EventEmitter.

const myEmitter = new MyEmitter()

myEmitter.on('event', function(a, b) {
   console.log(a, b, this);
});
myEmitter.emit('event', 'a', 'b');

console打?。?       a b MyEmitter {
         domain: null,
         _events: { event: [Function] },
         _eventsCount: 1,
         _maxListeners: undefine   //最大監(jiān)聽器數(shù)目為10個(gè)
       }

使用ES6的箭頭函數(shù)作為監(jiān)聽器,此時(shí)this不再指向EventEmitter實(shí)例

const myEmitter = new MyEmitter();

myEmitter.on('event', (a, b) => {
  console.log(a, b, this);
   // 打印: a b {}
});
myEmitter.emit('event', 'a', 'b');

3、異步與同步

EventListener會(huì)按照監(jiān)聽器的順序同步的調(diào)用所有監(jiān)聽器,所以可確保事件的正確順序且避免競(jìng)爭(zhēng)條件或邏輯錯(cuò)誤。監(jiān)聽器函數(shù)可以使用 setImmediate()或者process.nextTick() 方法切換到異步操作模式:

const myEmitter = new MyEmitter();

myEmitter.on('event', (a, b) => {
  setImmediate(() => {
    console.log('這個(gè)是異步發(fā)生的');
  });
});

myEmitter.emit('event', 'a', 'b');

4、只處理事件一次

當(dāng)使用eventEmitter.on()方法注冊(cè)監(jiān)聽器的時(shí)候,監(jiān)聽器會(huì)在每次觸發(fā)命名事件時(shí)被調(diào)用。

const myEmitter = new MyEmitter();
let m = 0;
myEmitter.on('event', () => {
  console.log(++m);
});
myEmitter.emit('event');
// 打印: 1
myEmitter.emit('event');
// 打印: 2

使用eventEmitter.once() 方法時(shí)可以注冊(cè)一個(gè)對(duì)于特定事件最多被調(diào)用一次的監(jiān)聽器。 當(dāng)事件被觸發(fā)時(shí),監(jiān)聽器會(huì)被注銷,然后再調(diào)用。

const myEmitter = new MyEmitter();
let m = 0;
myEmitter.once('event', () => {
  console.log(++m);
});
myEmitter.emit('event');
// 打印: 1
myEmitter.emit('event');
// 忽略

5、錯(cuò)誤事件

當(dāng)EventEmitter實(shí)例中發(fā)生錯(cuò)誤時(shí),將觸發(fā)一個(gè) 'error' 事件。如果 EventEmitter 沒有為 'error' 事件注冊(cè)至少一個(gè)監(jiān)聽器,則當(dāng) 'error' 事件觸發(fā)時(shí),會(huì)拋出錯(cuò)誤、打印堆棧跟蹤、且退出 Node.js 進(jìn)程。故而最佳實(shí)踐為,應(yīng)該始終為'error'事件注冊(cè)監(jiān)聽器。

const myEmitter = new MyEmitter()

myEmitter.on('error', (err) = > {
  console.error('have error')
})

myEmitter.emit('error',new Error( 'whoop!' ))
//打印: have error
最后編輯于
?著作權(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)容

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