精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

面試官:說說Node中的EventEmitter? 如何實現(xiàn)一個EventEmitter?

開發(fā) 前端
我們了解到,Node采用了事件驅(qū)動機制,而EventEmitter就是Node實現(xiàn)事件驅(qū)動的基礎。

[[404630]]

本文轉(zhuǎn)載自微信公眾號「JS每日一題」,作者灰灰。轉(zhuǎn)載本文請聯(lián)系JS每日一題公眾號。

一、是什么

我們了解到,Node采用了事件驅(qū)動機制,而EventEmitter就是Node實現(xiàn)事件驅(qū)動的基礎

在EventEmitter的基礎上,Node幾乎所有的模塊都繼承了這個類,這些模塊擁有了自己的事件,可以綁定/觸發(fā)監(jiān)聽器,實現(xiàn)了異步操作

Node.js 里面的許多對象都會分發(fā)事件,比如 fs.readStream 對象會在文件被打開的時候觸發(fā)一個事件

這些產(chǎn)生事件的對象都是 events.EventEmitter 的實例,這些對象有一個 eventEmitter.on() 函數(shù),用于將一個或多個函數(shù)綁定到命名事件上

二、使用方法

Node的events模塊只提供了一個EventEmitter類,這個類實現(xiàn)了Node異步事件驅(qū)動架構(gòu)的基本模式——觀察者模式

在這種模式中,被觀察者(主體)維護著一組其他對象派來(注冊)的觀察者,有新的對象對主體感興趣就注冊觀察者,不感興趣就取消訂閱,主體有更新的話就依次通知觀察者們

基本代碼如下所示:

  1. const EventEmitter = require('events'
  2.  
  3. class MyEmitter extends EventEmitter {} 
  4. const myEmitter = new MyEmitter() 
  5.  
  6. function callback() { 
  7.     console.log('觸發(fā)了event事件!'
  8. myEmitter.on('event', callback) 
  9. myEmitter.emit('event'
  10. myEmitter.removeListener('event', callback); 

通過實例對象的on方法注冊一個名為event的事件,通過emit方法觸發(fā)該事件,而removeListener用于取消事件的監(jiān)聽

關(guān)于其常見的方法如下:

  • emitter.addListener/on(eventName, listener) :添加類型為 eventName 的監(jiān)聽事件到事件數(shù)組尾部
  • emitter.prependListener(eventName, listener):添加類型為 eventName 的監(jiān)聽事件到事件數(shù)組頭部
  • emitter.emit(eventName[, ...args]):觸發(fā)類型為 eventName 的監(jiān)聽事件
  • emitter.removeListener/off(eventName, listener):移除類型為 eventName 的監(jiān)聽事件
  • emitter.once(eventName, listener):添加類型為 eventName 的監(jiān)聽事件,以后只能執(zhí)行一次并刪除
  • emitter.removeAllListeners([eventName]):移除全部類型為 eventName 的監(jiān)聽事件

三、實現(xiàn)過程

通過上面的方法了解,EventEmitter是一個構(gòu)造函數(shù),內(nèi)部存在一個包含所有事件的對象

  1. class EventEmitter { 
  2.     constructor() { 
  3.         this.events = {}; 
  4.     } 

其中events存放的監(jiān)聽事件的函數(shù)的結(jié)構(gòu)如下:

  1.   "event1": [f1,f2,f3], 
  2.   "event2": [f4,f5], 
  3.   ... 

然后開始一步步實現(xiàn)實例方法,首先是emit,第一個參數(shù)為事件的類型,第二個參數(shù)開始為觸發(fā)事件函數(shù)的參數(shù),實現(xiàn)如下:

  1. emit(type, ...args) { 
  2.     this.events[type].forEach((item) => { 
  3.         Reflect.apply(item, this, args); 
  4.     }); 

當實現(xiàn)了emit方法之后,然后實現(xiàn)on、addListener、prependListener這三個實例方法,都是添加事件監(jiān)聽觸發(fā)函數(shù),實現(xiàn)也是大同小異

  1. on(type, handler) { 
  2.     if (!this.events[type]) { 
  3.         this.events[type] = []; 
  4.     } 
  5.     this.events[type].push(handler); 
  6.  
  7. addListener(type,handler){ 
  8.     this.on(type,handler) 
  9.  
  10. prependListener(type, handler) { 
  11.     if (!this.events[type]) { 
  12.         this.events[type] = []; 
  13.     } 
  14.     this.events[type].unshift(handler); 

緊接著就是實現(xiàn)事件監(jiān)聽的方法removeListener/on

  1. removeListener(type, handler) { 
  2.     if (!this.events[type]) { 
  3.         return
  4.     } 
  5.     this.events[type] = this.events[type].filter(item => item !== handler); 
  6.  
  7. off(type,handler){ 
  8.     this.removeListener(type,handler) 

最后再來實現(xiàn)once方法, 再傳入事件監(jiān)聽處理函數(shù)的時候進行封裝,利用閉包的特性維護當前狀態(tài),通過fired屬性值判斷事件函數(shù)是否執(zhí)行過

  1. once(type, handler) { 
  2.     this.on(type, this._onceWrap(type, handler, this)); 
  3.   } 
  4.  
  5.   _onceWrap(type, handler, target) { 
  6.     const state = { fired: false, handler, type , target}; 
  7.     const wrapFn = this._onceWrapper.bind(state); 
  8.     state.wrapFn = wrapFn; 
  9.     return wrapFn; 
  10.   } 
  11.  
  12.   _onceWrapper(...args) { 
  13.     if (!this.fired) { 
  14.       this.fired = true
  15.       Reflect.apply(this.handler, this.target, args); 
  16.       this.target.off(this.type, this.wrapFn); 
  17.     } 
  18.  } 

完整代碼如下:

  1. class EventEmitter { 
  2.     constructor() { 
  3.         this.events = {}; 
  4.     } 
  5.  
  6.     on(type, handler) { 
  7.         if (!this.events[type]) { 
  8.             this.events[type] = []; 
  9.         } 
  10.         this.events[type].push(handler); 
  11.     } 
  12.  
  13.     addListener(type,handler){ 
  14.         this.on(type,handler) 
  15.     } 
  16.  
  17.     prependListener(type, handler) { 
  18.         if (!this.events[type]) { 
  19.             this.events[type] = []; 
  20.         } 
  21.         this.events[type].unshift(handler); 
  22.     } 
  23.  
  24.     removeListener(type, handler) { 
  25.         if (!this.events[type]) { 
  26.             return
  27.         } 
  28.         this.events[type] = this.events[type].filter(item => item !== handler); 
  29.     } 
  30.  
  31.     off(type,handler){ 
  32.         this.removeListener(type,handler) 
  33.     } 
  34.  
  35.     emit(type, ...args) { 
  36.         this.events[type].forEach((item) => { 
  37.             Reflect.apply(item, this, args); 
  38.         }); 
  39.     } 
  40.  
  41.     once(type, handler) { 
  42.         this.on(type, this._onceWrap(type, handler, this)); 
  43.     } 
  44.  
  45.     _onceWrap(type, handler, target) { 
  46.         const state = { fired: false, handler, type , target}; 
  47.         const wrapFn = this._onceWrapper.bind(state); 
  48.         state.wrapFn = wrapFn; 
  49.         return wrapFn; 
  50.     } 
  51.  
  52.     _onceWrapper(...args) { 
  53.         if (!this.fired) { 
  54.             this.fired = true
  55.             Reflect.apply(this.handler, this.target, args); 
  56.             this.target.off(this.type, this.wrapFn); 
  57.         } 
  58.     } 

測試代碼如下:

  1. const ee = new EventEmitter(); 
  2.  
  3. // 注冊所有事件 
  4. ee.once('wakeUp', (name) => { console.log(`${name} 1`); }); 
  5. ee.on('eat', (name) => { console.log(`${name} 2`) }); 
  6. ee.on('eat', (name) => { console.log(`${name} 3`) }); 
  7. const meetingFn = (name) => { console.log(`${name} 4`) }; 
  8. ee.on('work', meetingFn); 
  9. ee.on('work', (name) => { console.log(`${name} 5`) }); 
  10.  
  11. ee.emit('wakeUp''xx'); 
  12. ee.emit('wakeUp''xx');         // 第二次沒有觸發(fā) 
  13. ee.emit('eat''xx'); 
  14. ee.emit('work''xx'); 
  15. ee.off('work', meetingFn);        // 移除事件 
  16. ee.emit('work''xx');           // 再次工作 

參考文獻

 

  • http://nodejs.cn/api/events.html#events_class_eventemitter
  • https://segmentfault.com/a/1190000015762318
  • https://juejin.cn/post/6844903781230968845
  • https://vue3js.cn/interview

 

責任編輯:武曉燕 來源: JS每日一題
相關(guān)推薦

2022-04-12 08:09:22

Nodejs前端面試題

2024-03-28 10:37:44

IoC依賴注入依賴查找

2024-08-22 10:39:50

@Async注解代理

2024-03-05 10:33:39

AOPSpring編程

2021-05-20 08:34:03

CDN原理網(wǎng)絡

2025-04-08 00:00:00

@AsyncSpring異步

2025-10-20 04:00:00

2025-11-11 09:25:19

2024-03-14 14:56:22

反射Java數(shù)據(jù)庫連接

2024-07-31 08:28:37

DMAIOMMap

2024-12-06 07:00:00

2024-02-29 16:49:20

volatileJava并發(fā)編程

2024-09-20 08:36:43

零拷貝數(shù)據(jù)傳輸DMA

2024-03-22 06:56:24

零拷貝技術(shù)數(shù)據(jù)傳輸數(shù)據(jù)拷貝

2024-08-29 16:30:27

2024-08-12 17:36:54

2021-06-07 09:41:48

NodeBuffer 網(wǎng)絡協(xié)議

2021-06-08 08:33:23

NodeStream數(shù)據(jù)

2021-06-10 07:51:07

Node.js循環(huán)機制

2025-02-28 00:00:00

點贊
收藏

51CTO技術(shù)棧公眾號

免费观看不卡av| av影片在线| 国产精品一卡二| 久久青草精品视频免费观看| 一二三不卡视频| 日韩制服一区| 亚洲一卡二卡三卡四卡无卡久久| 26uuu亚洲综合色| 蜜臀久久99精品久久久久久宅男| 国产不卡一二三| 国产精品伦一区二区| 亚洲最新在线观看| 四虎一区二区| 视频在线观看你懂的| 精品一区中文字幕| 日韩av毛片网| 日本免费在线播放| 91精品秘密在线观看| 日韩精品在线观看一区二区| 亚洲一区二区图片| 成人在线黄色| 日韩欧美在线第一页| 可以在线看黄的网站| 亚洲精品传媒| 久久人人97超碰com| 99r国产精品视频| 中文字幕制服诱惑| 久久国产66| 国产+人+亚洲| 欧产日产国产v| 日韩极品一区| 一本色道久久综合狠狠躁篇的优点| av在线天堂网| 亚洲高清在线一区| 3751色影院一区二区三区| 日本精品免费在线观看| av在线视屏| 亚洲综合色自拍一区| 少妇熟女一区二区| 99re热久久这里只有精品34| 久久久久国产成人精品亚洲午夜| 久久本道综合色狠狠五月| 亚洲精品成人电影| 高清在线观看日韩| 97超级碰碰| 国产熟女精品视频| 国产精品一区一区| 亚洲xxxx在线| 成人av手机在线| av在线免费观看网址| 欧美三区不卡| 欧美成人性生活| 1024手机在线视频| 欧美 日韩 国产一区二区在线视频 | 国产精品视频第一页| 免费看欧美女人艹b| 国产精品福利在线观看网址| 日韩久久久久久久久久| 男男视频亚洲欧美| 国产日韩精品视频| 国产欧美一区二区三区视频在线观看| 激情久久久久久久久久久久久久久久| 国产欧美日韩中文| 国产女同91疯狂高潮互磨| 极品少妇xxxx精品少妇偷拍| 91中文在线观看| 国产黄色片av| 不卡的av电影| 日韩一区二区在线观看视频| 熟妇人妻va精品中文字幕| 桃花岛成人影院| 欧美在线短视频| 嫩草视频免费在线观看| 玖玖玖电影综合影院| 精品人在线二区三区| www国产视频| 亚洲三级精品| 中文国产成人精品| 国产性xxxx| 亚洲黄色免费| 国产精品扒开腿做| 国产又粗又猛视频| 成人国产在线观看| 日本免费高清一区二区| 三区四区在线视频| 一区二区三区在线观看动漫| 色欲av无码一区二区人妻| 99久久伊人| 精品人在线二区三区| 国产中年熟女高潮大集合| 色棕色天天综合网| 99国产欧美另类久久久精品| 亚洲伊人久久综合| 午夜性色福利视频| 中文字幕亚洲不卡| 少妇无码av无码专区在线观看 | 中文字幕一区二区三区在线视频| 久久久久亚洲精品| 这里只有精品9| 成人国产精品视频| 一本一本久久a久久精品综合妖精| 伊人影院在线视频| 欧美午夜在线观看| 水蜜桃av无码| 亚洲大全视频| 国产成人精品最新| 欧美一级在线免费观看| 国产精品久久久久影院色老大| 青青草国产精品视频| **国产精品| 国产视频精品一区二区三区| 中文字幕av久久爽av| 免费在线一区观看| 国产中文一区二区| 伊人福利在线| 欧美日韩一区二区三区免费看| 国模无码视频一区| 欧美精品国产| 成人h猎奇视频网站| 九色国产在线观看| 黄网动漫久久久| 久久黄色一级视频| 国产精品99一区二区三区| 热99精品里视频精品| 人妻va精品va欧美va| 亚洲欧美成aⅴ人在线观看| www.日日操| 校园春色另类视频| 久久久视频精品| 精品人妻一区二区三区麻豆91| 国产精品看片你懂得| 久草精品在线播放| 欧美日本成人| 热99精品里视频精品| 天天综合在线视频| 亚洲成人久久影院| 日本人妻一区二区三区| 国产一级一片免费播放放a| 欧美日韩国产亚洲一区| 91久久久久久久久久久| 77导航福利在线| 欧美日韩国产影片| 成人黄色短视频| 久久精品国产秦先生| 亚洲精品自在在线观看| 欧美aaa级| 日韩三级影视基地| 99精品在线视频观看| 亚洲精品大片www| 91视频免费入口| 欧美激情在线| 国产成人精品福利一区二区三区| 欧美videossex| 亚洲成av人片在线观看香蕉| 国产精品6666| 久久一夜天堂av一区二区三区| 欧美黄网站在线观看| 九一亚洲精品| 国产精品综合不卡av| 男人天堂久久久| 欧美一二三四区在线| 久久久久久欧美精品se一二三四| 国产999精品久久| 日韩xxxx视频| 亚州综合一区| 国产精品爽黄69天堂a| 黄视频在线观看网站| 日韩欧美不卡在线观看视频| 国产在线视频99| 91在线视频官网| 妺妺窝人体色www在线观看| 欧美一区二区三区高清视频| 国产日韩在线免费| 黄色大片在线| 亚洲女人天堂av| 中文字幕av久久爽| 亚洲自拍与偷拍| 超碰97在线资源站| 日本美女一区二区三区| 国产日本欧美在线| 九九热播视频在线精品6| 日本成人黄色片| 蜜桃视频网站在线观看| 亚洲高清色综合| 无码人妻久久一区二区三区 | 亚洲国产精品va在线看黑人| 亚洲黄色免费观看| 自拍偷拍欧美激情| 影音先锋黄色资源| 麻豆成人综合网| 老子影院午夜伦不卡大全| 国产亚洲一区二区三区不卡| 成人在线激情视频| 一个人看的www视频在线免费观看| 国产亚洲人成网站在线观看| 亚洲av无码乱码国产精品久久| 色综合久久久久综合体| 亚洲天堂黄色片| 久久久久久久久久电影| 好吊操视频这里只有精品| 日韩一区精品视频| 国产精品视频二| 成人黄色小视频| 精品国产乱码久久久久久郑州公司| 日韩成人亚洲| 午夜免费日韩视频| 成人影欧美片| 一区二区国产精品视频| 国产成人自拍一区| 在线不卡中文字幕| 国产免费av一区| 一片黄亚洲嫩模| 99久久99久久精品免费| 91论坛在线播放| 国产免费无码一区二区| 精品一区二区三区欧美| 国产主播在线看| 亚洲一级黄色| 青青在线免费视频| 婷婷伊人综合| 一级特黄录像免费播放全99| 久久不见久久见国语| 精品国产一区二区三区久久久久久| 精品视频91| 国产日韩欧美影视| 99久久婷婷国产综合精品首页| 日本成人在线视频网址| 深夜av在线| 97视频国产在线| 久久香蕉一区| 欧美成人黑人xx视频免费观看| 在线观看精品一区二区三区| 日韩av在线资源| 精品人妻久久久久一区二区三区| 欧美日本韩国一区二区三区视频 | 日韩一级黄色片| 亚洲中文一区二区三区| 欧美少妇性性性| 波多野结衣网站| 色综合天天综合网国产成人综合天| 日韩精品一卡二卡| 亚洲成人精品一区| 国产在线一二区| 亚洲一级二级在线| 国产性一乱一性一伧一色| 亚洲综合久久久久| 国产一级片免费观看| 亚洲一区二区四区蜜桃| 成人免费看片98| 五月天激情综合| 国产嫩bbwbbw高潮| 色欧美日韩亚洲| 中文在线最新版天堂| 欧美日韩成人综合| 国产又大又黄又爽| 日韩你懂的在线播放| 亚洲va久久久噜噜噜无码久久| 日韩免费在线观看| 欧美熟妇另类久久久久久不卡 | 97久久精品午夜一区二区| 视频一区在线| 精品国产日本| 国产欧美日韩视频在线| 亚洲一区二区不卡视频| 亚洲欧美日韩高清在线| 国产精品久久久久9999爆乳| 中文精品视频| 男女爽爽爽视频| 极品美女销魂一区二区三区 | 91精品91久久久中77777老牛| 亚洲欧美日韩在线观看a三区 | 狠狠色丁香久久婷婷综| 中文字幕一区二区在线观看视频| 国产一区二区伦理| 精品人妻一区二区三区日产| 久久影院视频免费| 少妇的滋味中文字幕bd| 亚洲精品免费电影| 国产又黄又猛又粗又爽| 欧美日韩精品欧美日韩精品| 午夜精品一二三区| 亚洲欧洲视频在线| caopeng在线| 欧美亚洲视频一区二区| www.精品国产| 国产精品夜夜夜一区二区三区尤| 欧美美乳视频| 妺妺窝人体色www看人体| 久久国产精品99国产| 在线播放免费视频| 久久久久高清精品| 久久久夜色精品| 在线中文字幕一区二区| 亚洲国产成人在线观看| 一区二区三区视频免费| 美女航空一级毛片在线播放| 国产精品国产三级国产aⅴ9色| 久久伦理中文字幕| 日韩精品伦理第一区| 伊人久久久大香线蕉综合直播| 性欧美videossex精品| jizz一区二区| 久久国产波多野结衣| 色综合视频一区二区三区高清| jlzzjlzzjlzz亚洲人| 中文字幕欧美在线| 另类激情视频| 国产一区二区在线观看免费播放| 久久美女视频| 成年人免费在线播放| 粉嫩蜜臀av国产精品网站| 美女福利视频网| 色噜噜夜夜夜综合网| 理论片中文字幕| 久久91精品国产| www.久久爱.com| 五月天久久综合网| 玖玖视频精品| 法国伦理少妇愉情| 亚洲超碰精品一区二区| 国产手机精品视频| 久久精品99久久久香蕉| 国产精成人品2018| 欧洲一区二区在线观看| 国产精品入口| 成人性生活免费看| 天天综合色天天综合| 精品人妻一区二区三区浪潮在线| 久久久av网站| 在线欧美激情| 在线精品日韩| 久久狠狠亚洲综合| 黄色裸体一级片| 欧美熟乱第一页| 视频三区在线| 国产精品视频xxxx| 日韩在线观看一区 | 国产成人免费视频网站视频社区 | 国产精品jizz视频| 欧美日韩亚洲一区三区| 欧美丰满熟妇bbb久久久| 亚洲精品成人悠悠色影视| av中文字幕观看| 欧美情侣性视频| 成功精品影院| 国产伦精品一区二区三区四区视频_| 丁香桃色午夜亚洲一区二区三区| 欧美另类视频在线观看| 亚洲精品在线免费观看视频| 超碰中文在线| 久久综合一区二区三区| 久久黄色网页| 91l九色lporny| 欧美日韩aaa| 在线免费观看a视频| 高清国产一区| 国产模特精品视频久久久久| 超碰97在线资源站| 在线观看免费亚洲| 蜜芽在线免费观看| 国产福利不卡| 香蕉久久夜色精品国产| 国产91丝袜美女在线播放| 欧美日韩一区不卡| 黄色美女视频在线观看| 久久99精品久久久久久秒播放器| 欧美一级视频| 欧美性x x x| 日韩美女在线视频| 亚洲欧美小说色综合小说一区| 日韩欧美三级电影| 国产自产v一区二区三区c| 日韩成年人视频| 在线丨暗呦小u女国产精品| 精品中文在线| 18禁男女爽爽爽午夜网站免费| 国产精品色婷婷久久58| www.国产精品视频| 青草青草久热精品视频在线网站 | www.桃色av嫩草.com| 国内精品久久久久久影视8| 精品freesex老太交| 日本高清免费观看| 日韩欧美在线视频日韩欧美在线视频| aiai在线| 黑人另类av| 久久精品国产秦先生| 日本中文字幕网| 日韩在线观看av| 狠狠v欧美ⅴ日韩v亚洲v大胸| 一本色道久久加勒比精品| 免费a在线观看| 亚洲a∨日韩av高清在线观看| 亚洲国产导航| 一本一本久久a久久| 亚洲国产精品人人爽夜夜爽| 日韩欧乱色一区二区三区在线| 日韩小视频在线播放| 亚洲欧美日韩一区二区三区在线观看 | 中文字幕无码日韩专区免费|