node.js如何自定義實(shí)現(xiàn)一個(gè)EventEmitter
前言
最近做了商品批發(fā)的需求,需要針對(duì)不同的商戶選擇對(duì)應(yīng)的批發(fā)商品回顯到原來(lái)的界面。由于該項(xiàng)目的代碼是公司古董級(jí)別(這種代碼都是程序猿的痛),解決問(wèn)題的時(shí)候都是小心翼翼的。為了避免這種問(wèn)題減少外部依賴,手動(dòng)封裝事件派發(fā)的函數(shù)。
一、是什么
我們了解到,Node采用了事件驅(qū)動(dòng)機(jī)制,而EventEmitter就是Node實(shí)現(xiàn)事件驅(qū)動(dòng)的基礎(chǔ)
在EventEmitter的基礎(chǔ)上,Node幾乎所有的模塊都繼承了這個(gè)類,這些模塊擁有了自己的事件,可以綁定/觸發(fā)監(jiān)聽(tīng)器,實(shí)現(xiàn)了異步操作
Node.js 里面的許多對(duì)象都會(huì)分發(fā)事件,比如 fs.readStream 對(duì)象會(huì)在文件被打開(kāi)的時(shí)候觸發(fā)一個(gè)事件
這些產(chǎn)生事件的對(duì)象都是 events.EventEmitter 的實(shí)例,這些對(duì)象有一個(gè) eventEmitter.on() 函數(shù),用于將一個(gè)或多個(gè)函數(shù)綁定到命名事件上
二、nodejs中EventEmitter使用方法
Node的events模塊只提供了一個(gè)EventEmitter類,這個(gè)類實(shí)現(xiàn)了Node異步事件驅(qū)動(dòng)架構(gòu)的基本模式——觀察者模式
在這種模式中,被觀察者(主體)維護(hù)著一組其他對(duì)象派來(lái)(注冊(cè))的觀察者,有新的對(duì)象對(duì)主體感興趣就注冊(cè)觀察者,不感興趣就取消訂閱,主體有更新的話就依次通知觀察者們
const EventEmitter = require('events') class MyEmitter extends EventEmitter {} const myEmitter = new MyEmitter() function callback() { console.log('觸發(fā)了event事件!') } myEmitter.on('event', callback) myEmitter.emit('event') myEmitter.removeListener('event', callback);
三、實(shí)現(xiàn)過(guò)程
基本代碼如下所示:
//事件派發(fā)機(jī)制 (function() { var EventDispatcher = function() { var EventDispatcherClosure = function() { }; EventDispatcherClosure.prototype = { /** * 注冊(cè)事件 * @param {Object} key * @param {Object} fn */ on: function(key, fn) { //獲取當(dāng)前的事件對(duì)象 var curEvents = this._getCurEvents(key); //先檢查該事件是否已經(jīng)注冊(cè)過(guò)了 var flag = false; for (var i = 0, len = curEvents.length; i < len; i++) { if (curEvents[i].name == fn.name) { //已經(jīng)出現(xiàn)過(guò)了,以最新注冊(cè)的函數(shù)為主 flag = true; curEvents[i] = fn; break; } } if (!flag) { curEvents[curEvents.length] = fn; } this._register(key, curEvents); }, /** * 派發(fā)事件 * @param {Object} key * @param {Object} data */ dispatch: function(key) { //獲取當(dāng)前的事件對(duì)象 var curEvents = this._getCurEvents(key); var shouldDispatch = true; for (var i = 0, len = curEvents.length; shouldDispatch && i < len; i++) { try { //獲取參數(shù) var args = []; for (var j = 1, len1 = arguments.length; j < len1; j++) { args.push(arguments[j]); } shouldDispatch = curEvents[i].apply({}, args); } catch (e) { shouldDispatch = false; } } return shouldDispatch; }, remove: function(key) { if (this._getCurEvents(key)) { delete EventDispatcherClosure.events[key]; } }, /** * 根據(jù)key獲取事件列表 * @param {Object} key */ _getCurEvents: function(key) { return EventDispatcherClosure.events[key] || []; }, /** * 注冊(cè)時(shí)間 * @param {Object} key * @param {Object} events */ _register: function(key, events) { EventDispatcherClosure.events[key] = events; }, }; EventDispatcherClosure.events = {}; return { create: function() { return new EventDispatcherClosure(); } }; }; window.EventDispatcher = new EventDispatcher().create(); })();
首先定義一個(gè)全局變量的匿名函數(shù),然后將全局變量掛在window上面,這樣可以讓我們?cè)陂_(kāi)發(fā)過(guò)程中的調(diào)用。在匿名函數(shù)的原型鏈上面添加事件分發(fā)、事件監(jiān)聽(tīng)、事件刪除等方法。
事件分發(fā)的調(diào)用
EventDispatcher.dispatch("test", obj)
事件監(jiān)聽(tīng)
EventDispatcher.on("test", function callback(obj) { })
事件刪除
EventDispatcher.on("test")
代碼封裝的比較簡(jiǎn)單
到此這篇關(guān)于node.js如何自定義實(shí)現(xiàn)一個(gè)EventEmitter的文章就介紹到這了,更多相關(guān)node實(shí)現(xiàn)EventEmitter內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
node.js博客項(xiàng)目開(kāi)發(fā)手記
本篇文章給大家總結(jié)了node.js博客項(xiàng)目開(kāi)發(fā)的相關(guān)步驟以及知識(shí)點(diǎn)分享,有興趣的朋友參考下。2018-03-03Node.js設(shè)置CORS跨域請(qǐng)求中多域名白名單的方法
這篇文章主要介紹了Node.js設(shè)置CORS跨域請(qǐng)求中多域名白名單的方法,文中通過(guò)示例代碼介紹的非常詳細(xì),相信對(duì)大家具有一定的參考價(jià)值,需要的朋友們下面來(lái)一起看看吧。2017-03-03Node.js web 應(yīng)用如何封裝到Docker容器中
這篇文章主要介紹了Node.js web 應(yīng)用如何封裝到Docker容器中,幫助大家更好的學(xué)習(xí)node.js和使用docker容器,感興趣的朋友可以了解下2020-09-09Express框架實(shí)現(xiàn)簡(jiǎn)單攔截器功能示例
這篇文章主要介紹了Express框架實(shí)現(xiàn)簡(jiǎn)單攔截器功能,結(jié)合實(shí)例形式分析了express框架攔截器相關(guān)功能與使用方法,需要的朋友可以參考下2023-05-05Node.js操作redis實(shí)現(xiàn)添加查詢功能
Redis 是一個(gè)基于內(nèi)存的鍵(key)值(value)類型的數(shù)據(jù)結(jié)構(gòu)存儲(chǔ)容器,它既可以完全工作在內(nèi)存中,也可以持久化存儲(chǔ)。當(dāng) Redis 工作于持久化模式時(shí),可以將它當(dāng)作一個(gè)非關(guān)系型數(shù)據(jù)庫(kù)使用。2017-05-05詳解基于Node.js的HTTP/2 Server實(shí)踐
HTTP/2目前已經(jīng)逐漸的在各大網(wǎng)站上開(kāi)始使用,這篇文章主要介紹了詳解基于Node.js的HTTP/2 Server實(shí)踐,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-05-05詳解node如何讓一個(gè)端口同時(shí)支持https與http
眾所周知node是一個(gè)高性能的web服務(wù)器,使用它可以很簡(jiǎn)單的創(chuàng)建一個(gè)http或https的服務(wù)器。這篇文章主要介紹了詳解node如何讓一個(gè)端口同時(shí)支持https與http2017-07-07