nodeJS之事件events

来源:http://www.cnblogs.com/xiaohuochai/archive/2017/06/05/6947386.html
-Advertisement-
Play Games

[1]EventEmitter [2]方法 [3]設置 [4]事件 ...


前面的話

  events模塊是node的核心模塊,幾乎所有常用的node模塊都繼承了events模塊,比如http、fs等。本文將詳細介紹nodeJS中的事件機制

 

EventEmitter

  多數 Node.js 核心 API 都是採用慣用的非同步事件驅動架構,其中某些類型的對象(稱為觸發器)會周期性地觸發命名事件來調用函數對象(監聽器)。例如,一個net.Server對象會在每次有新連接時觸發一個事件;一個 fs.ReadStream 會在文件被打開時觸發一個事件;一個 stream會在數據可讀時觸發事件。

【EventEmitter】

  EventEmitter 類由 events 模塊定義和開放的,所有能觸發事件的對象都是 EventEmitter 類的實例

var EventEmitter = require('events');
/*
{ [Function: EventEmitter]
  EventEmitter: [Circular],
  usingDomains: false,
  defaultMaxListeners: [Getter/Setter],
  init: [Function],
  listenerCount: [Function] }
 */
console.log(EventEmitter);

  events模塊的EventEmitter屬性指向該模塊本身

var events = require('events');
console.log(events.EventEmitter === events);//true

  EventEmitter是一個構造函數,可以用來生成事件發生器的實例emitter

var EventEmitter = require('events');
var emitter = new EventEmitter();
/*
EventEmitter {
  domain: null,
  _events: {},
  _eventsCount: 0,
  _maxListeners: undefined }
 */
console.log(emitter);

 

方法

【emitter.emit(eventName[, ...args])】

eventName <any>
...args <any>

  該方法按監聽器的註冊順序,同步地調用每個註冊到名為eventName事件的監聽器,並傳入提供的參數。如果事件有監聽器,則返回true,否則返回false

var EventEmitter = require('events');
var emitter = new EventEmitter();
emitter.on('test1',function(){});
console.log(emitter.emit('test1'));//true
console.log(emitter.emit('test2'));//false

【emitter.on(eventName, listener)】

  該方法用於添加listener函數到名為eventName的事件的監聽器數組的末尾

eventName <any> 事件名
listener <Function> 回調函數

  [註意]不會檢查listener是否已被添加。多次調用並傳入相同的eventName和listener會導致listener被添加與調用多次

var EventEmitter = require('events');
var emitter = new EventEmitter();
emitter.on('test',function(){
    console.log(1);
});
emitter.on('test',function(){
    console.log(2);
});
emitter.emit('test');//1 2

  該方法返回一個 EventEmitter 引用,可以鏈式調用

var EventEmitter = require('events');
var emitter = new EventEmitter();
emitter.on('test',function(){
    console.log(1);
}).on('test',function(){
    console.log(2);
});
emitter.emit('test');//1 2

【emitter.addListener(eventName, listener)】

  emitter.on(eventName, listener) 的別名

var EventEmitter = require('events');
var emitter = new EventEmitter();
emitter.addListener('test',function(){
    console.log(1);
});
emitter.emit('test');//1

【emitter.prependListener()】

  與on()方法不同,prependListener()方法可用於將事件監聽器添加到監聽器數組的開頭

var EventEmitter = require('events');
var emitter = new EventEmitter();
emitter.on('test',function(){
    console.log(1);
}).prependListener('test',function(){
    console.log(2);
});
emitter.emit('test');//2 1

【emitter.once(eventName, listener)】

  該方法添加一個單次 listener 函數到名為 eventName 的事件。 下次觸發 eventName 事件時,監聽器會被移除,然後調用

eventName <any> 事件名
listener <Function> 回調函數
var EventEmitter = require('events');
var emitter = new EventEmitter();
emitter.on('test',function(){
    console.log(1);
}).once('test',function(){
    console.log(2);
});
emitter.emit('test');//1 2
emitter.emit('test');//1

【emitter.prependOnceListener() 】

  該方法用於將事件監聽器添加到監聽器數組開頭。下次觸發eventName事件時,監聽器會被移除,然後調用

var EventEmitter = require('events');
var emitter = new EventEmitter();
emitter.on('test',function(){
    console.log(1);
}).prependOnceListener('test',function(){
    console.log(2);
});
emitter.emit('test');//2 1
emitter.emit('test');//1

【emitter.removeAllListeners([eventName])】

eventName <any>

  移除全部或指定 eventName 的監聽器,返回一個 EventEmitter 引用,可以鏈式調用

  [註意]在代碼中移除其他地方添加的監聽器是一個不好的做法,尤其是當 EventEmitter 實例是其他組件或模塊(如 socket 或文件流)創建的

var EventEmitter = require('events');
var emitter = new EventEmitter();
emitter.on('test',function(){
    console.log(1);
}).removeAllListeners('test');
emitter.emit('test');//''

【emitter.removeListener(eventName, listener)】

eventName <any>
listener <Function>

  從名為 eventName 的事件的監聽器數組中移除指定的 listener

var EventEmitter = require('events');
var emitter = new EventEmitter();
function show(){
    console.log(1);
}
emitter.on('test',show).removeListener('test',show);
emitter.emit('test');//''

  [註意]removeListener最多只會從監聽器數組裡移除一個監聽器實例。如果任何單一的監聽器被多次添加到指定eventName的監聽器數組中,則必須多次調用removeListener才能移除每個實例

var EventEmitter = require('events');
var emitter = new EventEmitter();
function show(){
    console.log(1);
}
emitter.on('test',show).on('test',show).removeListener('test',show);
emitter.emit('test');//'1'

  [註意]一旦一個事件被觸發,所有綁定到它的監聽器都會按順序依次觸發。這意味著,在事件觸發後、最後一個監聽器完成執行前,任何 removeListener() 或 removeAllListeners() 調用都不會從 emit() 中移除它們。 隨後的事件會像預期的那樣發生

  因為監聽器是使用內部數組進行管理的,所以調用它會改變在監聽器被移除後註冊的任何監聽器的位置索引。 雖然這不會影響監聽器的調用順序,但意味著由 emitter.listeners() 方法返回的監聽器數組副本需要被重新創建

var EventEmitter = require('events');
var emitter = new EventEmitter();
function show1(){
    console.log(1);
    emitter.removeListener('test',show2);
}
function show2(){
    console.log(2);
}
emitter.on('test',show1).on('test',show2);
emitter.emit('test');//1 2
emitter.emit('test');//1

 

設置

【emitter.eventNames()】

  返回一個列出觸發器已註冊監聽器的事件的數組。 數組中的值為字元串或符號

var EventEmitter = require('events');
var emitter = new EventEmitter();
emitter.addListener('test1',function(){console.log(1);});
emitter.addListener('test2',function(){console.log(2);});
console.log(emitter.eventNames());//[ 'test1', 'test2' ]

【emitter.listenerCount(eventName)】

eventName <any> 正在被監聽的事件名

  返回正在監聽名為 eventName 的事件的監聽器的數量

var EventEmitter = require('events');
var emitter = new EventEmitter();
emitter.addListener('test',function(){console.log(1);});
emitter.addListener('test',function(){console.log(1);});
console.log(emitter.listenerCount('test'));//2

【emitter.listeners(eventName)】

eventName <any>

  返回名為 eventName 的事件的監聽器數組的副本

var EventEmitter = require('events');
var emitter = new EventEmitter();
emitter.addListener('test',function(){console.log(1);});
emitter.addListener('test',function(){console.log(2);});
console.log(emitter.listeners('test'));//[ [Function], [Function] ]
emitter.listeners('test')[0]();//1

【emitter.getMaxListeners()】

  返回 EventEmitter 當前的最大監聽器限制值

var EventEmitter = require('events');
var emitter = new EventEmitter();
console.log(emitter.getMaxListeners());//10

【emitter.setMaxListeners(n)】

  預設情況下,如果為特定事件添加了超過 10 個監聽器,則 EventEmitter 會列印一個警告。 此限制有助於尋找記憶體泄露。 但是,並不是所有的事件都要被限為 10 個。 emitter.setMaxListeners() 方法允許修改指定的 EventEmitter 實例的限制。 值設為 Infinity(或 0)表明不限制監聽器的數量。返回一個 EventEmitter 引用,可以鏈式調用

var EventEmitter = require('events');
var emitter = new EventEmitter();
emitter.on('a',function(){}).on('a',function(){}).on('a',function(){}).on('a',function(){}).on('a',function(){}).on('a',function(){}).on('a',function(){}).on('a',function(){}).on('a',function(){}).on('a',function(){}).on('a',function(){});
/*
Warning: Possible EventEmitter memory leak detected. 11 a listeners added. Use emitter.setMaxListeners() to increase limit
 */
var EventEmitter = require('events');
var emitter = new EventEmitter();
emitter.setMaxListeners(11);
emitter.on('a',function(){}).on('a',function(){}).on('a',function(){}).on('a',function(){}).on('a',function(){}).on('a',function(){}).on('a',function(){}).on('a',function(){}).on('a',function(){}).on('a',function(){}).on('a',function(){});

【EventEmitter.defaultMaxListeners】
  每個事件預設可以註冊最多10個監聽器。單個EventEmitter實例的限制可以使用emitter.setMaxListeners(n)方法改變。所有EventEmitter實例的預設值可以使用EventEmitter.defaultMaxListeners屬性改變

  [註意]設置 EventEmitter.defaultMaxListeners 要謹慎,因為會影響所有EventEmitter 實例,包括之前創建的。因而,調用 emitter.setMaxListeners(n) 優先於 EventEmitter.defaultMaxListeners

var EventEmitter = require('events');
var emitter = new EventEmitter();
EventEmitter.defaultMaxListeners = 11;
emitter.on('a',function(){}).on('a',function(){}).on('a',function(){}).on('a',function(){}).on('a',function(){}).on('a',function(){}).on('a',function(){}).on('a',function(){}).on('a',function(){}).on('a',function(){}).on('a',function(){});

 

事件

【'newListener' 事件】

eventName <any> 要監聽的事件的名稱
listener <Function> 事件的句柄函數

  EventEmitter 實例會在一個監聽器被添加到其內部監聽器數組之前觸發自身的 'newListener' 事件

  註冊了 'newListener' 事件的監聽器會傳入事件名與被添加的監聽器的引用。事實上,在添加監聽器之前觸發事件有一個微妙但重要的副作用: 'newListener' 回調中任何額外的被註冊到相同名稱的監聽器會在監聽器被添加之前被插入 

var EventEmitter = require('events');
var emitter = new EventEmitter();
emitter.on('newListener',function(){
    console.log(2);
})
emitter.on('test',function(){
    console.log(1);
})

emitter.emit('test');//2 1
var EventEmitter = require('events');
var emitter = new EventEmitter();
emitter.on('test',function(){
    console.log(1);
})
emitter.on('newListener',function(){
    console.log(2);
})
emitter.emit('test');//1

【'removeListener' 事件】

eventName <any> 事件名
listener <Function> 事件句柄函數

  'removeListener' 事件在 listener 被移除後觸發

var EventEmitter = require('events');
var emitter = new EventEmitter();
function show(){
    console.log(1);
}
emitter.on('removeListener',function(){
    console.log(2);//2
})
emitter.on('test',show).removeListener('test',show);
var EventEmitter = require('events');
var emitter = new EventEmitter();
function show(){
    console.log(1);
}
emitter.on('test',show).removeListener('test',show);
emitter.on('removeListener',function(){
    console.log(2);//''
})
var EventEmitter = require('events');
var emitter = new EventEmitter();
function show(){
    console.log(1);
}
emitter.removeListener('test',show);
emitter.on('removeListener',function(){
    console.log(2);//''
})

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 創建一個子類對象會不會創建父類對象? 不會,只創建了一個子類對象,但是往父類對象的構造方法里傳了子類對象的地址;給子類對象初始化的時候,調用了父類的構造方法。 證明: 結果: A==366712642 B==366712642 如果子類對象創建的同時也創建了一個父類對象,那麼父類和子類構造方法中th ...
  • 在互動式環境中輸入: 如下圖: 還是在互動式環境中: 圖片展示: 這種反射機制的用字元串來操作類的屬性和方法的三個函數並不常用。編寫框架等特殊項目是採用到。 但這時用戶仍然可以通過w._water來訪問實例屬性,封裝的不好,也不會自動檢查數據是不是浮點型,不好。 怎麼解決? 用@屬性.setter ...
  • 一.概念 反射就是把Java的各種成分映射成相應的Java類。 Class類的構造方法是private,由JVM創建。 反射是java語言的一個特性,它允程式在運行時(註意不是編譯的時候)來進行自我檢查並且對內部的成員進行操作。例如它允許一個java的類獲取他所有的成員變數和方法並且顯示出來。Jav ...
  • 單鏈表: * 1.鏈表可以是一種有序或無序的列表 * 2.鏈表的內容通常存儲在記憶體中分散的為止 * 3.鏈表由節點組成,每一個節點具有相同的結構 * 4.節點分為數據域和鏈域,數據域存放節點內容,鏈域存放下一個節點的指針 package myLinkList; public class MyLink ...
  • 在js中常常要求對時間的輸出格式進行格式化,比如 2017-01-01 10:10,比較麻煩的是月,日,小時,分。它們的格式一般要求兩位,如果小於10的話需要在前邊補0,當然這算不上什麼問題,可以通過判斷完成,比如 day = day>10?day:("0"+day);但是這樣的代碼寫多了總想寫點有 ...
  • Array 數組類 除了 Object 之外, Array 類型恐怕是ECMAScript 中最常用的類型了。而且,ECMAScript 中的數組與其他多數語言中的數組有著相當大的區別。 雖然 ECMAScript 數組與其他語言中的數組都是數據的有序列表,但與其他語言不同的是,ECMAScript ...
  • 1、Make Images Mobile Responsive 用處: 使圖片適配你的頁面寬度。 操作: 給圖片添加 .img-responsive class屬性。 2、Center Text with Bootstrap 用處: 使文本居中。 操作: 給文本添加 center-text clas ...
  • <!DOCTYPE html><html lang="en"><head> <meta charset="UTF-8"> <title>Title</title></head> <body><script>/** splice(start, deleteCount, data1, data2, da ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...