国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁(yè) > 開(kāi)發(fā) > JS > 正文

JavaScript EventEmitter 背后的秘密 完整版

2024-05-06 16:43:24
字體:
來(lái)源:轉(zhuǎn)載
供稿:網(wǎng)友

什么是 Event Emitter?

Event emitter 聽(tīng)起來(lái)只是觸發(fā)一個(gè)事件,這個(gè)事件任何東西都能監(jiān)聽(tīng)。

想象一下這樣的場(chǎng)景,在你的異步代碼中,去“呼叫”一些事件的發(fā)生,以及讓你其他部分都要聽(tīng)到你的“呼叫”并且注冊(cè)他們的想法。

為了不同的目的,對(duì)于 Event Emitter 模式有大量不同的實(shí)現(xiàn),但是基本的想法是為了給一個(gè)框架提供事件的管理以及能夠去訂閱他們。

在這里,我們的目標(biāo)創(chuàng)建屬于我們自己的 Event Emitter 去理解背后的秘密。所以,讓我們看一下下面的代碼是怎么工作的。

let input = document.querySelector("input[type="text"]");let button = document.querySelector("button");let h1 = document.querySelector("h1");button.addEventListener("click", () => {  emitter.emit("event:name-changed", { name: input.value });});let emitter = new EventEmitter();emitter.subscribe("event:name-changed", data => {  h1.innerHTML = `Your name is: ${data.name}`;});

讓我們開(kāi)始。

class EventEmitter {  constructor() {    this.events = {};  }}

我們先創(chuàng)建一個(gè) EventEmiiter 類(lèi)以及初始化 events 空對(duì)象屬性。這個(gè) events 屬性的目的是為了存儲(chǔ)我們的事件集合,這個(gè) events 對(duì)象使用事件名當(dāng)做 key,用訂閱者集合當(dāng)做 value。(可以把每個(gè)訂閱者看作是一個(gè)函數(shù))。

訂閱函數(shù)

subscribe(eventName, fn) {  if (!this.events[eventName]) {    this.events[eventName] = [];  }  this.events[eventName].push(fn);}

這個(gè)訂閱函數(shù)獲取事件名稱(chēng),在我們之前的例子中,它是 "event:name-changed" 以及傳入一個(gè)回調(diào),當(dāng)有人調(diào)用 emit(或尖叫)事件的時(shí)候調(diào)用回調(diào)。

JavaScript 函數(shù)的優(yōu)點(diǎn)之一是函數(shù)是第一對(duì)象,所以我們能像之前我們的訂閱方法一樣,通過(guò)函數(shù)作為另一個(gè)函數(shù)的參數(shù)。

如果未注冊(cè)這個(gè)事件,我們需要在第一次為它設(shè)置一個(gè)初始值,事件名稱(chēng)作為 key 以及初始化一個(gè)空數(shù)組賦值給它,然后我們將函數(shù)放入這個(gè)數(shù)組,以便我們想通過(guò) emit 去調(diào)用這個(gè)事件。

調(diào)用函數(shù)

emit(eventName, data) {  const event = this.events[eventName];  if (event) {    event.forEach(fn => {      fn.call(null, data);    });  }}

這個(gè)調(diào)用函數(shù)接受事件名,這個(gè)事件名是我們想“呼叫”的名稱(chēng),以及我們想傳遞給這個(gè)事件的數(shù)據(jù)。如果在我們的 events 中存在這個(gè)事件,我們將帶上數(shù)據(jù)循環(huán)調(diào)用所有訂閱的方法。

使用上面的代碼能做我們所說(shuō)的全部的事情。但我們?nèi)匀挥幸粋€(gè)問(wèn)題。當(dāng)我們不再需要它們的時(shí)候,我們需要一種方法來(lái)取消注冊(cè)這些訂閱,因?yàn)槿绻悴贿@樣做,將造成內(nèi)存泄漏。

讓我們來(lái)解決這個(gè)問(wèn)題,通過(guò)在訂閱函數(shù)中返回一個(gè)取消注冊(cè)的方法。

subscribe(eventName, fn) {  if (!this.events[eventName]) {    this.events[eventName] = [];  }  this.events[eventName].push(fn);  return () => {    this.events[eventName] = this.events[eventName].filter(eventFn => fn !== eventFn);  }}

因?yàn)?JavaScript 函數(shù)是第一對(duì)象,你能在一個(gè)函數(shù)中返回一個(gè)函數(shù)。因此現(xiàn)在我們能調(diào)用這個(gè)取消注冊(cè)函數(shù),如下:

let unsubscribe = emitter.subscribe("event:name-changed", data => console.log(data));unsubscribe();

當(dāng)我們調(diào)用取消注冊(cè)函數(shù)的時(shí)候,我們刪除的功能依賴(lài)于對(duì)訂閱函數(shù)集合的篩選方法(Array filter)。

和內(nèi)存泄露說(shuō)再見(jiàn)!
注:相關(guān)教程知識(shí)閱讀請(qǐng)移步到JavaScript/Ajax教程頻道。

發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 务川| 门源| 珲春市| 衡山县| 连山| 江西省| 昌平区| 马鞍山市| 定边县| 惠来县| 鄂伦春自治旗| 琼结县| 浦江县| 太白县| 东源县| 顺义区| 军事| 南城县| 洪雅县| 应用必备| 盱眙县| 东乡县| 金坛市| 安达市| 京山县| 南安市| 东台市| 西昌市| 常德市| 天台县| 大方县| 高淳县| 高碑店市| 盐源县| 晋州市| 长泰县| 太仆寺旗| 光泽县| 阿克苏市| 吴堡县| 祁门县|