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)

相关内容

热门资讯

全球首套,中天科技交付220k... IT之家 5 月 11 日消息,据中天科技集团消息,近日,中天科技交付全球首套 220kV 3500...
原创 北... 北京业主刚以488万的价格卖掉了自己的二手房,三天后宁愿付违约金,也要把房子拿回来。转手加价70多万...
“硬科技”场内基金频发溢价风险... 【导读】硬科技场内基金频发溢价风险提示 中国基金报记者天心 日前,多只聚焦海内外半导体芯片方向的场内...
伯希和再闯港股陷更名争议,CE... 5月8日,国内户外运动品牌伯希和(PELLIOT)再度向港交所递交上市申请,中金公司与中信证券担任联...
一季度货币政策报告明确:引导隔... 5月11日,人民银行披露一季度中国货币政策执行报告,指出下一步将引导隔夜利率在政策利率水平附近运行,...
科博会观察|能源转型的“下半场... 今年4月,光伏龙头隆基绿能发布“全栈隆基LONGi ONE”光储融合战略,这场发布会背后是公司对能源...
“茅台魔咒”失灵了?沪指站上4... 11日,沪指走出“八连阳”,站上4200点,创下自2015年6月26日以来的收盘点位新高。 板块方...
沪指涨0.94%站上4200点... 扬子晚报网5月11日讯(记者 范晓林)截至午盘,沪指站上4200点,创业板指大涨突破3900点,为2...
ETF周评:4200点之前,“... “五一”假期后的首个交易周(5月6日至5月8日),A股虽仅有短短三个交易日,却展现出强劲的做多动能。...
集智达GNS-2446主板赋能... 当前医疗自助终端面临四大行业痛点:多任务并发算力瓶颈;外设兼容集成难题;数据安全合规压力;复杂环境稳...
动荡市场中锚定稀缺确定性,新能... 3月以来,美伊冲突导致全球能源价格出现大幅波动。整体上看,本轮地缘冲突的复杂性和影响深度远超以往,加...
CFA协会:未来金融人才需具备... 由特许金融分析师协会(CFA协会)、北京市金融发展促进中心共同主办的2026第五届中国未来金融分析师...
最熟悉的“国民理财神器”,让你... 1万元放进余额宝,一天收益只有0.24元,连个鸡蛋都买不起。这不是某个冷门产品,而是那个曾创下6.7...
Circle从贝莱德等机构融资... 来源:环球市场播报 核心要点 Circle 互联网集团在其全新 Arc 区块链关联代币预售中融资...
张尧浠:美伊局势变数不断 金价... 来源:市场资讯 5月11日:黄金市场上周:国际黄金伦敦金触底回升收涨,再度收取垂线止跌看涨形态,但上...
中澳企业拓展新能源合作 来源:人民日报 2026年澳大利亚智慧能源展日前在悉尼国际会议中心举行。当前,中东局势引发全球能源...
七类技能培训“套路”曝光 中消... 记者今天(11日)从中消协获悉,近年来,各类技能培训迅速扩张,新型培训模式不断涌现,部分经营者借助新...
每日收评沪指涨超1%站上420... 财联社5月11日讯,市场全天震荡走强,沪指站上4200点,创业板指大涨突破3900点,为2015年6...
A股三大指数集体上涨:沪指站上... 观点网讯:5月11日,A股三大指数集体上涨,截至收盘,上证指数涨1.08%站上4200点,深证成指涨...