nodejs篇 内置模块events 常用api
admin
2024-02-15 14:38:12
0

文章目录

    • 前提
    • 引入
    • 注册事件的两种方式
      • on
      • addListener
    • 和vue事件绑定做对比
    • removeListener(eventName, listener)
    • events参数的传递
    • eventEmitter.once()
    • 自带的两种特殊事件
      • newListener 事件
      • removeListener事件
    • removeAllListeners([eventName])
    • 不常用的api
      • MaxListeners

前提

如果你接触过vue,并学习和使用过里面的事件绑定和事件触发,以及子组件给父组件传参的$emit方式,你可能会觉得,本章的内容异常熟悉。

如果你觉得nodejs官方文档给的api太多,不知道哪些重要,请看下去,否则请前往
https://nodejs.org/dist/latest-v10.x/docs/api/events.html

如果你认为官方文档全是英文,不太容易学,并且不在乎nodejs中文网站版本过低的问题,请前往
https://www.nodeapp.cn/events.html

引入

// events 本身是一个构造函数,或者说是一个类,可以构建一个实例对象。
const Events = require("events");

注册事件的两种方式

on

const Events = require("events");
// events 本身是一个构造函数,或者说是一个类,可以构建一个实例对象。
const myEvent = new  Events()// 实例对象上的on方法,用来注册该实例对象的dx事件
myEvent.on('dx',() => {console.log('触发了dx事件')
})setTimeout(() => {// 实例对象的emit方法,用来触发该实例对象上注册的dx事件myEvent.emit('dx')
}, 1000);

同一个事件对象,同一个事件名称,可以多次注册

const Events = require("events");
const myEvent = new  Events()
myEvent.on('click',() => {console.log('click被触发了')
})myEvent.on('click',() => {console.log('click又被触发了')
})myEvent.emit('click')
// click被触发了
// click又被触发了

addListener

addListener的使用方式与on完全一致

const Events = require("events");
// events 本身是一个构造函数,或者说是一个类,可以构建一个实例对象。
const myEvent = new  Events()// 实例对象上的on方法,用来注册该实例对象的dx事件
myEvent.addListener('dx',() => {console.log('触发了dx事件')
})setTimeout(() => {// 实例对象的emit方法,用来触发该实例对象上注册的dx事件myEvent.emit('dx')
}, 1000);

和vue事件绑定做对比

// 组件本身是一个构造函数,或者说是一个类,可以构建一个实例对象,形成一个组件实例。
components: {myEvent: MyEvent
}// 组件实例上的v-on方法,用来注册该组件实例的dx事件
 {console.log('触发了dx事件')
}" />// 在myEvent组件内部某个时候
setTimeout(() => {// 实例对象的emit方法,用来触发该实例对象上注册的dx事件this.$emit('dx')
}, 1000);

你会发现,events的设计模式和vue事件注册触发的设计模式几乎一致,只是写法不同而已,后面就不作对比了,继续介绍events,但越往后面,越觉得它们是一摸一样的,换汤不换药。

removeListener(eventName, listener)

移除事件对象的事件监听,与js的removeEventListener基本一致,需要确保注册和移除时填入的参数能够分别完全相等,否则将无法移除

"dx" === "dx" // true
handle === handle  // true
const Events = require("events");
const myEvent = new  Events()const handle = () => {console.log("dx被触发了");
};myEvent.on("dx", handle);myEvent.emit("dx"); // dx被触发了
myEvent.removeListener("dx", handle);
myEvent.emit("dx"); // 不会执行handle

events参数的传递

const Events = require("events");// events 本身是一个构造函数,或者说是一个类,可以构建一个实例对象。
const myEvent = new  Events()// 实例对象上的on方法,用来注册该实例对象的事件
myEvent.on('dx',(age,name) => {console.log(`${name}今年${age}岁`) // yx今年18岁
})setTimeout(() => {// 实例对象的emit方法,用来触发该实例对象上注册的某一事件// 第一个参数是事件名称,后面的参数都是传递给回调函数的参数myEvent.emit('dx',18,'yx')
}, 1000);

eventEmitter.once()

eventEmitter.once() 用来注册只能被执行一次的事件,once使用的方式与on一致,这有点像vue里面的事件修饰符once,比如 @click.once

const Events = require("events");
// events 本身是一个构造函数,或者说是一个类,可以构建一个实例对象。
const myEvent = new  Events()// 实例对象上的on方法,用来注册该实例对象的事件
myEvent.once('dx',(age,name) => {console.log(`${name}今年${age}岁`)
})setTimeout(() => {// 实例对象的emit方法,用来触发该实例对象上注册的某一事件myEvent.emit('dx',18,'yx') // yx今年18岁myEvent.emit('dx',20,'dx') // 未执行
}, 1000);

自带的两种特殊事件

newListener 事件

newListener是一个特殊的事件名称,它能监听到事件对象每一次有事件被注册,你也许有许多通用的逻辑需要处理.

newListener事件的回调,只会在被绑定的事件对象(myEvent),有事件绑定时才会触发,当myEvent触发事件时,不会触发newListener的回调。

const Events = require("events");
const myEvent = new  Events()
// event 就是事件的名称 handle就是对应的回调函数
myEvent.on('newListener',(event,handle) => {console.log(event)// ... 你还可以写一些其它的通用逻辑handle()
})myEvent.on('click',() => {console.log('click被注册了')
})myEvent.on('dx',() => {console.log('dx被注册了')
})// click
// click被注册了
// dx
// dx被注册了

即使在事件对象上重复注册同一个事件,newListener的回调每一次都会触发

const Events = require("events");
const myEvent = new  Events()myEvent.on('newListener',(event,handle) => {console.log(event)// ... 你还可以写一些其它的通用逻辑handle()
})myEvent.on('click',() => {console.log('click被注册了')
})myEvent.on('click',() => {console.log('click又被注册了')
})// click
// click被注册了
// click
// click又被注册了

removeListener事件

removeListener 也是一个特殊的事件名称,它能监听到事件对象每一次有事件被成功移除,你也许有许多通用的逻辑需要处理.

const Events = require("events");
const myEvent = new  Events()myEvent.on("removeListener", (event, handle) => {handle();console.log(`${event}被移除了`)
});const handle = () => {console.log("dx事件回调被执行了");
};myEvent.on("dx", handle);myEvent.removeListener("dx", handle);// dx事件回调被执行了
// dx被移除了

如果移除某一事件失败,removeListener事件的回调是不会触发的

const Events = require("events");
const myEvent = new  Events()myEvent.on("removeListener", (event, handle) => {handle();console.log(`${event}被移除了`)
});const handle = () => {console.log("dx事件回调被执行了");
};// myEvent.on("dx", handle);myEvent.removeListener("dx", handle); 
// dx未被注册是不可能移除成功的,所以没有任何代码被console.log

removeAllListeners([eventName])

移除全部或指定 eventName 的监听器。世界瞬间安静了。

const Events = require("events");
const myEvent = new  Events()
myEvent.removeAllListeners() // 移除所有绑定的事件

移除dx和click事件所有绑定的回调

myEvent.removeAllListeners(['dx','click'])

不常用的api

MaxListeners

默认的MaxListeners是10,意味着同一个事件对象上的同一个事件名,不能被绑定超过10个回调,如果超过会报一个警告,用以防止内存泄漏。
这个警告不是在你触发(click)事件才会出现,而是在你绑定的超过的时候就会出现。

const Events = require("events");
const myEvent = new  Events()myEvent.on('click',() => {console.log('click被触发了1')
})myEvent.on('click',() => {console.log('click又被触发了2')
})myEvent.on('click',() => {console.log('click被触发了3')
})myEvent.on('click',() => {console.log('click又被触发了4')
})myEvent.on('click',() => {console.log('click被触发了5')
})myEvent.on('click',() => {console.log('click又被触发了6')
})myEvent.on('click',() => {console.log('click被触发了7')
})myEvent.on('click',() => {console.log('click又被触发了8')
})
myEvent.on('click',() => {console.log('click被触发了9')
})myEvent.on('click',() => {console.log('click又被触发了10')
})
myEvent.on('click',() => {console.log('click被触发了11')
})myEvent.on('click',() => {console.log('click又被触发了12')
})
myEvent.on('click',() => {console.log('click被触发了13')
})myEvent.emit('click')
// click被触发了1
// click又被触发了2
// click被触发了3
// click又被触发了4
// click被触发了5
// click又被触发了6
// click被触发了7
// click又被触发了8
// click被触发了9
// click又被触发了10
// click被触发了11
// click又被触发了12
// click被触发了13
// (node:24276) MaxListenersExceededWarning: Possible EventEmitter memory leak detected. 11 click listeners added to [EventEmitter]. Use emitter.setMaxListeners() to increase limit
// (Use `node --trace-warnings ...` to show where the warning was created)

可以通过setMaxListeners(n)修改这个值

const Events = require("events");
const myEvent = new  Events()myEvent.setMaxListeners(20)

相关内容

热门资讯

斗金订购APP贵金属期货投资被...   斗金订购APP的投资者被广告宣传给诱导,注册就送什么现金,然后充值返现金卷等等这些宣传方式,都是...
哈易购APP非法期货交易欺骗投...   哈易购APP宣传可做白银铂金贵金属订购交易,但实际上并没有取得相关交易资质!哈易购APP本质上就...
消息称百度旗下昆仑芯瞄准500... 6 月 29 日消息,据《The Information》昨日援引知情人士消息,百度旗下 AI 芯片...
打造夏日消费新场景 第35届北... 北京商报讯(记者 翟枫瑞)6月29日消息,第35届北京国际燕京啤酒文化节新闻发布会在京举行。本届啤酒...
社保基金持仓数据出炉,一季度增... 最近各大上市公司一季度财报都公开了,咱们国家社保基金的持仓数据也全部曝光。目前社保拿着比亚迪价值44...
36氪首发 | 海思、中兴团队... 作者 | 乔钰杰 编辑 | 袁斯来 硬氪获悉,广州宸思通讯科技有限公司(以下简称“宸思科技”)近日完...
两天蒸发47亿市值!一纸税务通... 一纸税务通知书,能让一家百亿龙头两天蒸发47亿市值。 6月22日,北大荒(600598.SH)公告称...
SK海力士将投资1100万亿韩... SK集团会长崔泰源6月29日在韩国“三大重大计划”发布会上宣布,公司将投资1100万亿韩元扩大半导体...
两只A股,终止上市! 两家A股公司,即将摘牌。 6月29日,退市沪科(600608.SH)公告称,上海证券交易所将在202...
原创 M... 一家成立近十年的自动驾驶公司,在IPO时吸引了14家基石投资者认购近一半的发行股份,其中不乏奔驰、比...
基金忠言|国寿安保滤镜碎,三年... 图片来源:视觉中国 蓝鲸新闻6月29日讯(记者 祁和忠)保险系基金公司国寿安保总经理换人了。 6月2...
三星电机计划加码玻璃基板!相关... 6月29日,玻璃基板概念股午后有所回升, 华工科技(000988.SZ)逼近涨停, 彩虹股份(600...
拉萨海关持续壮大外贸经营主体 ...   新华网拉萨6月28日电(记者蒋梦辰)近日,记者从拉萨海关获悉,今年前5个月,西藏有进出口实绩的外...
机构:二季报临近,医药生物板块... 6月29日,华源证券发布了一篇医药生物行业的研究报告,报告指出,业绩期临近,产业链景气度有望再次迎来...
每日收评科创50放量涨超4.5... 财联社6月29日讯,三大指数全线收红,创业板指探底回升,科创50指数大涨4.61%。沪深两市成交额3...
6月多地土拍结构性升温:深圳单... 进入2026年6月,不少城市核心区地块集中诞生高溢价宗地,热度突出的城市包含深圳、杭州、长沙。 其中...
业绩炸裂!盛达资源半年预盈3.... 6月29日,贵金属矿山龙头盛达资源(000603.SZ)发布 2026 年半年度业绩预告,上半年业绩...
A股午后拉升三大股指收涨:半导... A股三大股指6月29日开盘涨跌互现。早盘沪强深弱,创指一度跌超2%。半导体午后拉升,带动两市上涨,沪...
原创 空... 前言 大家好,我是老金。 这几天,两幅极度割裂的画面放在一起,把我看笑了。 一边是在持续的热浪下,欧...