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

EventEmitter 的核心功能實(shí)現(xiàn)

開發(fā) 前端
EventEmitter 是 Nodejs 環(huán)境下才能使用的庫,所以不能直接用于瀏覽器環(huán)境的開發(fā)。所以我考慮自己實(shí)現(xiàn)一套邏輯,自己定制的話也容易根據(jù)實(shí)際情況的變動(dòng)做修改。

大家好,我是前端西瓜哥。

EventEmitter 是頻率較高的前端面試題。

EventEmitter 是 Nodejs 環(huán)境下才能使用的庫,所以不能直接用于瀏覽器環(huán)境的開發(fā)。所以我考慮自己實(shí)現(xiàn)一套邏輯,自己定制的話也容易根據(jù)實(shí)際情況的變動(dòng)做修改。

因此我決定了解一下 EventEmitter 的 API,并嘗試自己實(shí)現(xiàn)一套邏輯。

Nodejs 的 EventEmitter API

首先當(dāng)然是要了解需求,即 EventEmitter 的 API 使用。詳細(xì)使用方式請查閱 官方文檔,我這里只簡單敘述一些常用的 API。

const { EventEmitter, errorMonitor } = require('events');
// 創(chuàng)建事件觸發(fā)器實(shí)例
const emitter = new EventEmitter()
// on:注冊監(jiān)聽者函數(shù)。可以注冊多個(gè)監(jiān)聽函數(shù),
// 觸發(fā)事件后,會(huì)依次同步執(zhí)行,順序?yàn)榻壎〞r(shí)的順序。
// 別名為:addListener
emitter.on('event', function(a, b) => {
console.log('event emit!', a, b)
})
// once:注冊一個(gè)只會(huì)被執(zhí)行一次的函數(shù)
emitter.once('event', function() => {
console.log('event emit only once!')
})
// emit:觸發(fā)事件,可提供參數(shù)。
// 如果有對應(yīng)監(jiān)聽器函數(shù),會(huì)返回 true,否則返回 false
emitter.emit('event', 3, 4)
// 比較特別的是,如果沒有注冊 error 事件的監(jiān)聽者,
// 觸發(fā) error 時(shí),錯(cuò)誤不會(huì)被捕獲而直接報(bào)錯(cuò);
// 若注冊,則錯(cuò)誤會(huì)被捕獲
emitter.emit('error', new Error('whoops!'))
// event.errorMonitor 是一個(gè) Symbol
// 能夠在觸發(fā) error 事件時(shí),先執(zhí)行被綁定的監(jiān)視器函數(shù)
emitter.on(errorMonitor, err => {
console.log('error monitor')
});
// 移除指定監(jiān)聽器,別名為:removeListener
emitter.off(eventName, handler)
// 獲取注冊的事件名的數(shù)組形式
emitter.eventNames()
  • 監(jiān)聽者函數(shù)的 this 會(huì)指向 EventEmitter 實(shí)例。當(dāng)然你可以使用各種方法修改 this 的指向,如箭頭函數(shù)或 bind 方法。
  • 每次添加監(jiān)聽器時(shí),都會(huì)觸發(fā) newListener 事件,傳入的參數(shù)為事件名(eventName)和監(jiān)聽器函數(shù)(listener)。
  • 同樣,移除監(jiān)聽器時(shí),會(huì)觸發(fā) removeListener 事件。
  • emitter.prependListener():同 on,但會(huì)添加到監(jiān)聽器數(shù)組的開頭。
  • ...

API 很多,但我不打算實(shí)現(xiàn)了這么多,就只實(shí)現(xiàn)最常用的 on、emit、off。

實(shí)現(xiàn)

首先,我們知道不同的事件是有特定的 eventName(事件名)的,通過指定 eventName,我們才能綁定對應(yīng)的多個(gè)監(jiān)聽器(函數(shù)),才能觸發(fā)事件執(zhí)行綁定的這些監(jiān)聽器。

這時(shí)候,我們就涉及到數(shù)據(jù)結(jié)構(gòu)與算法的存儲問題了。因?yàn)榻Y(jié)構(gòu)和算法是相輔相成的,選擇不同的數(shù)據(jù)結(jié)構(gòu),使用的算法就會(huì)不同。

不同的數(shù)據(jù)結(jié)構(gòu)與算法的優(yōu)點(diǎn)的缺陷各不相同,比如空間復(fù)雜度上或時(shí)間復(fù)雜度上的效率不同。

listener 函數(shù)的存儲

那么如何存儲呢?常見的方法是使用哈希表,因?yàn)闀r(shí)間復(fù)雜度是 O(1),空間復(fù)雜度一般也不會(huì)太大。JavaScript 的對象本質(zhì)上就是哈希表。所以我們的存儲方式是:

this.hashMap = {
'event1': [listener1, listenr2],
'event2': [],
}

一些可擴(kuò)展的點(diǎn):

  • 哈希表的一個(gè)問題是:無序??梢酝ㄟ^額外使用一個(gè)數(shù)組來記錄添加 eventName 的記錄順序。這樣的話,實(shí)現(xiàn) emitter.eventNames() 可以拿到有序的事件數(shù)據(jù)。當(dāng)然這樣的需求比較少見,這里只是簡單提一下。
  • 如果要實(shí)現(xiàn) once(設(shè)置執(zhí)行一次就不再執(zhí)行的監(jiān)聽器函數(shù)),則需要對函數(shù)標(biāo)記,這時(shí)候可以考慮讓數(shù)組元素的格式改為 { listener: Listener, once: boolean },在觸發(fā)事件的時(shí)候,執(zhí)行監(jiān)聽器函數(shù)時(shí),將 once 值為 true 的監(jiān)聽器從數(shù)組中移除。
  • 可以改為鏈表實(shí)現(xiàn)存儲,這樣移除中間監(jiān)聽器時(shí),時(shí)間復(fù)雜度可以變成 O(1)。另外數(shù)組刪除元素的時(shí)間復(fù)雜度是 O(n)。但會(huì)引入實(shí)現(xiàn)上的復(fù)雜度,因?yàn)闆]有內(nèi)置的鏈表實(shí)現(xiàn),需要自己手動(dòng)實(shí)現(xiàn)一個(gè)沒有 BUG 的鏈表類。

on() 的實(shí)現(xiàn)

on() 的實(shí)現(xiàn),其實(shí)就是將監(jiān)聽器函數(shù)綁定到指定事件對應(yīng)的數(shù)組中。實(shí)現(xiàn)起來并不難,只要注意如果是第一次添加指定事件時(shí),要先初始化一個(gè)空數(shù)組即可。on 最后返回了 this,是為了實(shí)現(xiàn)鏈?zhǔn)秸{(diào)用。

class EventEmiter {
on(eventName, listener) {
if (!this.hashMap[eventName]) {
this.hashMap[eventName] = []
}
this.hashMap[eventName].push(listener)
return this
}
}

off() 的實(shí)現(xiàn)

off() 會(huì)根據(jù)傳入的事件名,找到對應(yīng)的監(jiān)聽器數(shù)組,從中移除指定監(jiān)聽器。同樣為了實(shí)現(xiàn)鏈?zhǔn)秸{(diào)用返回了 this。

class EventEmiter {
off(eventName, listener) {
const listeners = this.hashMap[eventName]
if (listeners && listeners.length > 0) {
const index = listeners.indexOf(listener)
if (index > -1) {
listeners.splice(index, 1)
}
}
return this
}
}

emit() 的實(shí)現(xiàn)

emit() 的實(shí)現(xiàn)很簡單,找到事件對應(yīng)的監(jiān)聽器,傳入?yún)?shù)依次執(zhí)行。如果事件沒有綁定監(jiān)聽器,返回 false。否則,返回 true。

class EventEmiter {
emit(eventName, ...args) {
const listeners = this.hashMap[eventName]
if (!listeners || listeners.length === 0) return false
listeners.forEach(listener => {
listener(...args)
})
return true
}
}

完整實(shí)現(xiàn)

雖然很突然,我這里給出的是 TypeScript 實(shí)現(xiàn),只要將類型聲明去掉就是 JavaScript 實(shí)現(xiàn)了。當(dāng)然下面代碼是做了簡單的單元測試的,大概是沒問題的。

源碼地址:

type EventName = string | symbol
type Listener = (...args: any[]) => void
class EventEmiter {
private hashMap: { [eventName: string]: Array<Listener> } = {}
on(eventName: EventName, listener: Listener): this {
const name = eventName as string
if (!this.hashMap[name]) {
this.hashMap[name] = []
}
this.hashMap[name].push(listener)
return this
}
emit(eventName: EventName, ...args: any[]): boolean {
const listeners = this.hashMap[eventName as string]
if (!listeners || listeners.length === 0) return false
listeners.forEach(listener => {
listener(...args)
})
return true
}
off(eventName: EventName, listener: Listener): this {
const listeners = this.hashMap[eventName as string]
if (listeners && listeners.length > 0) {
const index = listeners.indexOf(listener)
if (index > -1) {
listeners.splice(index, 1)
}
}
return this
}
}

因?yàn)閷ο蟛恢С?Symbol 作為索引,所以這里的實(shí)現(xiàn)做了類型的強(qiáng)轉(zhuǎn)。未來,TypeScript 可能會(huì)允許對象索引為 Symbol,Enum 等,但目前不行。

責(zé)任編輯:姜華 來源: 今日頭條
相關(guān)推薦

2023-08-04 14:31:43

Python核心項(xiàng)目

2010-01-27 17:38:58

Windows Emb

2009-11-18 13:11:29

PHP核心

2024-02-29 07:48:55

Python編程語言上下文管理器

2010-02-07 14:16:57

2024-04-23 00:00:00

SpringBoot監(jiān)聽器

2023-03-01 08:15:10

NginxNacos

2010-04-07 09:31:06

2024-08-01 08:45:17

2011-05-26 17:19:05

中間件

2023-10-10 14:56:27

物聯(lián)網(wǎng)智能建筑智能樓宇

2024-11-12 10:57:14

NumPyPython

2010-09-22 15:31:05

OracleSPARCSolaris

2013-02-28 14:52:29

VMware

2011-03-07 09:55:56

2021-07-14 11:07:56

AOPJDKCglib

2020-11-11 08:04:34

低代碼

2023-02-23 08:15:33

Spring異常處理機(jī)制

2019-08-15 10:29:35

物聯(lián)網(wǎng)卡物聯(lián)網(wǎng)平臺

2017-11-30 06:07:53

點(diǎn)贊
收藏

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

国产手机精品视频| 插我舔内射18免费视频| 色综合久久影院| 精品中文字幕一区二区小辣椒| 久久五月天色综合| 欧美在线一级片| 91p九色成人| 亚洲一区二区三区国产| 欧美日韩中文国产一区发布| 国产精品探花视频| 一区二区国产精品| 久久精品视频99| 黄色网址在线视频| 色综合视频一区二区三区44| 亚洲国产cao| 亚洲午夜精品国产| 亚洲欧美另类日韩| 免费一级欧美片在线观看| 色与欲影视天天看综合网| 好吊视频在线观看| 粉嫩av一区二区| 欧美日韩国产在线播放网站| 免费成人午夜视频| www免费在线观看| 久久久久国产成人精品亚洲午夜| 国产精品一区久久久| 国产在线观看免费av| 欧美限制电影| 亚洲精品一区久久久久久| 91大神免费观看| 四虎4545www精品视频| 午夜精品久久久久久久99樱桃| 一区二区三区四区欧美日韩| 国产三级电影在线观看| 99久久伊人网影院| 国产精品一区二| 精品乱子伦一区二区| 理论片日本一区| 国产精品美女久久久免费| 在线观看免费国产视频| 亚洲夜间福利| 欧美成人午夜影院| 91香蕉一区二区三区在线观看| 伊人久久大香线蕉无限次| 欧美精品一区二区三区蜜桃| 色婷婷综合在线观看| 国产亚洲欧美日韩精品一区二区三区 | 裸体xxxx视频在线| 99re这里都是精品| 国产欧美日本在线| 三级网站免费观看| 成人免费视频播放| 国产乱码精品一区二区三区日韩精品| 国产a级免费视频| 国内精品久久久久影院薰衣草| 国产精品成人观看视频国产奇米| 五月天婷婷激情| 免费视频一区| 国产91精品在线播放| 天天干天天操天天爱| 性色一区二区三区| 国产精品极品美女粉嫩高清在线| 加勒比在线一区| 日本欧美在线看| 国产男人精品视频| 国产精品伦一区二区三区| 国产专区欧美精品| 亚洲影院污污.| 精品国产av 无码一区二区三区| 毛片不卡一区二区| 96pao国产成视频永久免费| 国产免费一区二区三区免费视频| 国产一区美女在线| 国产精品国产精品国产专区不卡| 四虎精品一区二区三区| 26uuu精品一区二区| 日本精品一区二区三区视频| 色影院视频在线| 一区二区三区精品视频| 777777av| 中文字幕日本一区二区| 欧美肥妇毛茸茸| 日本wwwwwww| 综合亚洲色图| 日韩在线观看免费网站| 青草影院在线观看| 国产日韩1区| 国产精品视频免费观看www| 国产aⅴ爽av久久久久成人| 不卡视频免费播放| 日韩成人av网站| 日本亚洲精品| 亚洲成人av一区二区| 成人性视频欧美一区二区三区| 99久久久成人国产精品| 亚洲福利在线视频| 俄罗斯毛片基地| 国产一区二区三区四区三区四 | 亚洲欧洲日本国产| 中文字幕资源网在线观看| 欧美日韩色婷婷| 污视频网站观看| 巨人精品**| 久久精品国产成人精品| 可以在线观看av的网站| 精品一区二区三区不卡| 久久五月天婷婷| 国产激情视频在线观看| 欧美色另类天堂2015| 日本中文字幕在线不卡| 国产一区二区亚洲| 欧美激情视频一区二区| 中文字幕视频二区| 99re热视频精品| eeuss中文| 日韩av中字| 亚洲第一精品福利| 午夜爽爽爽男女免费观看| 先锋a资源在线看亚洲| 亚洲专区中文字幕| 国产女人在线观看| 精品国产电影一区| 九色91porny| 欧美日韩有码| 青青草国产精品一区二区| 精品国产av一区二区三区| 欧美极品另类videosde| 国产精品丝袜久久久久久消防器材| 亚洲精品无播放器在线播放| 亚洲欧洲日本专区| 日韩成人在线免费视频| 国产成人精品aa毛片| 椎名由奈jux491在线播放| 日韩在线免费| 精品亚洲一区二区三区| 日韩人妻无码一区二区三区99 | 91久久精品国产91久久| 国产三级在线看| 日本高清不卡视频| 午夜理伦三级做爰电影| 亚洲三级免费| 国产精品一区二区免费| 国产亚av手机在线观看| 日韩久久精品一区| 欧美日韩人妻精品一区二区三区| 韩国v欧美v日本v亚洲v| 亚洲国产精品日韩| 国产激情欧美| 精品国产一区二区三区久久| 亚洲特级黄色片| 国产精品入口麻豆原神| 制服丝袜综合网| 日韩精品免费一区二区三区| 国产精品视频专区| 日韩毛片久久久| 欧美人xxxx| 国产极品美女在线| 国内精品久久久久影院薰衣草| 特色特色大片在线| 亚洲日本一区二区三区在线| 欧美精品xxx| 四季av日韩精品一区| 岛国av在线不卡| 久久成人激情视频| 麻豆精品一区二区av白丝在线| 亚洲视频在线观看日本a| 在线播放成人| 欧美黄色成人网| 天堂在线视频网站| 日本久久一区二区三区| 女人裸体性做爰全过| 国产精品中文字幕欧美| 久久99中文字幕| 亚洲三级网页| 国产欧美一区二区三区久久 | 91精品国产亚洲| 国产精品毛片一区二区三区四区| 欧美日韩色一区| 免费一级黄色大片| 91欧美一区二区| 国产色视频在线播放| 综合久久十次| 蜜桃在线一区二区三区精品| 日韩在线激情| 久久久亚洲欧洲日产国码aⅴ| 三级在线播放| 在线播放/欧美激情| 日本熟伦人妇xxxx| 欧美高清在线一区二区| 老女人性生活视频| 久久精品1区| 日本三日本三级少妇三级66| 欧美在线导航| 91九色精品视频| 伊人久久精品一区二区三区| 久久精品电影网| 色天堂在线视频| 欧美一级淫片007| 欧美一区二区三区不卡视频| 亚洲免费观看视频| 欧美做受高潮6| 成人午夜av电影| 可以看污的网站| 亚洲一区不卡| 992tv快乐视频| 日韩大片在线播放| 激情视频一区二区| 欧美一区一区| 国产精品美女主播| 男人的天堂免费在线视频| 久久九九免费视频| 黄色在线观看网| 亚洲第一精品夜夜躁人人爽| 999国产精品视频免费| 一本久久a久久免费精品不卡| 免费日韩在线视频| 国产精品久久毛片| 免费看黄色的视频| gogo大胆日本视频一区| 色偷偷中文字幕| 久久99精品久久久久久动态图| 欧美视频第一区| 黑人一区二区| 九一免费在线观看| 久久影视一区| 亚洲精品美女久久7777777| 亚洲国产欧美日韩在线观看第一区| 亚洲综合精品一区二区| 日韩av黄色| 国产精品久久久久久一区二区| 理论片午夜视频在线观看| 久久久久国产精品www| 91中文在线| 久久av.com| 高潮毛片在线观看| 久久天天躁夜夜躁狠狠躁2022| 国产黄在线看| 亚洲人成在线观看网站高清| 欧洲毛片在线| 亚洲美女av电影| 日韩黄色影片| 亚洲色图美腿丝袜| 国产三级视频在线| 中文字幕av一区中文字幕天堂 | 欧美成人免费va影院高清| 麻豆视频在线| 久久久999精品视频| 国产黄色小视频在线| 免费91麻豆精品国产自产在线观看| 国产原创视频在线观看| 精品久久久999| 在线中文字幕第一页| 欧美日韩成人在线视频| 欧美高清另类hdvideosexjaⅴ| 欧美激情久久久| www.51av欧美视频| 欧美国产视频日韩| 超碰激情在线| 欧美在线视频a| 日韩一级二级| 成人国产精品av| 无码国模国产在线观看| 国产精品免费在线| 香蕉久久精品| 午夜老司机精品| 久久久久亚洲| 麻豆tv在线播放| 视频一区国产视频| 国产高清av片| 成人av免费网站| 国产sm调教视频| 最新日韩在线视频| 欧美毛片在线观看| 欧美性猛交xxxx偷拍洗澡| 影音先锋黄色网址| 日韩免费电影网站| 日本一区高清| 久久精品国产欧美亚洲人人爽| 高清电影在线免费观看| 日本91av在线播放| 成人免费91| 精选一区二区三区四区五区| 色777狠狠狠综合伊人| 久草视频这里只有精品| 久久在线精品| 2025中文字幕| 国产婷婷色一区二区三区| 亚洲精品一区二区三区在线播放| 亚洲一区在线观看免费观看电影高清 | 亚洲成av人影院| 中文字幕永久在线视频| 精品伦理精品一区| 国产在线你懂得| 欧美激情在线视频二区| 日韩大尺度黄色| 99免费在线视频观看| 欧美精品第一区| 日韩精品久久一区二区| 老**午夜毛片一区二区三区 | 最新中文字幕免费| 精品91自产拍在线观看一区| 在线视频自拍| 日本精品久久久久久久| 午夜视频在线观看精品中文| 五月天丁香综合久久国产 | 国产精品入口免费软件| 丁香婷婷综合网| 亚洲欧美精品久久| 色域天天综合网| 日韩一级片免费看| 久久精品一本久久99精品| 欧美人体一区二区三区| 成人h视频在线观看| 日韩理论在线| 国产成人av影视| av电影天堂一区二区在线观看| 五月天色婷婷丁香| 91激情五月电影| 手机在线观看毛片| 欧美精品www| 香蕉大人久久国产成人av| 亚洲欧美日韩不卡一区二区三区| 亚洲尤物在线| 国产精品一级黄片| 亚洲一区二区三区视频在线播放 | 亚洲精品国产系列| 免费亚洲一区| 欧美 变态 另类 人妖| 亚洲高清免费视频| 国产成人三级一区二区在线观看一| 中文字幕久热精品视频在线| 黄色亚洲网站| 免费观看成人在线| 国产日韩综合| 老熟妇精品一区二区三区| 亚洲国产欧美在线人成| 性欧美8khd高清极品| 欧美成年人在线观看| 二区三区精品| 欧美日韩在线免费观看视频| 美女一区二区视频| 快灬快灬一下爽蜜桃在线观看| 色婷婷综合在线| fc2在线中文字幕| 国产精品视频久久久久| 欧美国产一级| www.污网站| 亚洲欧美日韩中文字幕一区二区三区| 一卡二卡三卡在线观看| 久久亚洲国产成人| 精品视频在线播放一区二区三区| 国产免费xxx| 高清av一区二区| 国产一级淫片a| 亚洲精品国产品国语在线| 国产精品25p| 欧洲一区二区在线| 日本视频在线一区| 国产精品免费人成网站酒店| 日韩一区二区电影在线| 国产啊啊啊视频在线观看| 久久影院理伦片| 美女脱光内衣内裤视频久久网站 | 大又大又粗又硬又爽少妇毛片| 一本色道综合亚洲| 2019中文字幕在线视频| 91九色综合久久| 精品av久久久久电影| 亚洲av无码一区二区三区观看| 日韩欧美在线视频免费观看| 99re在线视频| 91久久偷偷做嫩草影院| 伊人天天综合| 黄免费在线观看| 在线播放国产精品二区一二区四区| 三级资源在线| 麻豆一区区三区四区产品精品蜜桃| 日韩激情一二三区| wwwav国产| 亚洲欧洲成视频免费观看| 日韩五码电影| 亚洲美免无码中文字幕在线| 欧美高清在线一区| 亚洲精品综合网| 国产精品盗摄久久久| 欧美99在线视频观看| 美国黄色一级毛片| 91精品免费观看| 中文在线最新版地址| 91手机视频在线| 91一区在线观看| 国产精品午夜福利| 欧美亚洲国产视频| 永久91嫩草亚洲精品人人| 国产福利短视频| 日韩亚洲欧美中文三级| 欧美成人免费电影| www.一区二区.com| 亚洲国产精品av| 亚州精品国产精品乱码不99按摩|