内容简介:你有没有想过,为什么浏览器的 div 上可以绑定多个实际上这一切都是 EventEmitter 在背后做支持,它是 JavaScript 经典的事件驱动实现,现在我们来看下 Node.js 中是如何实现的。本文所说的监听事件在实现上都为函数,读者可以认为两者相等以方便阅读。
你有没有想过,为什么浏览器的 div 上可以绑定多个 onclick
事件,点击一下 div 可以触发全部的事件,jquery 的 .on()
, .off()
, one()
又是如何实现的?Node.js 事件驱动的原理是怎样的?
实际上这一切都是 EventEmitter 在背后做支持,它是 JavaScript 经典的事件驱动实现,现在我们来看下 Node.js 中是如何实现的。
本文所说的监听事件在实现上都为函数,读者可以认为两者相等以方便阅读。
因为原来的 Node 代码量比较多,为了方便演示,作者把本文的源代码示例中涉及数据验证,错误处理的部分删除,保留了主要内容。
准备:
- 源码一份: github.com/nodejs/node…
概览 EventEmitter
内部属性:
_events _eventsCount
主要方法:
emitter.addListener/on(eventName, listener) emitter.prependListener(eventName, listener) emitter.emit(eventName[, ...args]) emitter.removeListener/off(eventName, listener) emitter.once(eventName, listener) emitter.removeAllListeners([eventName])
正文
1. 初始化 init
_events
不存在时,使用 Object.create(null)
来初始化,并把 _eventsCount
设 0。
划重点 —— Object.create(null) 可以创建一个没有原型的对象。
为什么要用这种方法创建对象呢?开发者这么做的目的其实还是出于性能上的考虑,因为 EventEmitter 在 Node.js 中应用广泛,为节省服务器内存和执行速度上不必要的开销,肯定能省则省呗。
2. 添加事件绑定 addListener
首先判断 target
的 _events
是否存在,如果不存在则还是用 Object.create(null)
创建。
如果存在,触发 newListener
类型的事件。然后通过 event[type]
找到已经注册 type 类型的监听事件/监听事件数组,并存到 existing
中。
如果该事件值为 undefined
,则把直接把要注册的监听事件 listener
赋给不存在的事件。否则,更新事件数组(单一的 listener
要转为数组)。
注意 prepend 的使用,可以灵活地把 listener
添加到监听函数数组头部或尾部。
3. 事件添加到数组头部 prependListener
和 addListener 类似,但是prepend
为
true
。
2. 触发事件 emit
若 handler 不存在,直接返回 false。
若 handler 是一个函数,使用 Reflect
调用函数。如果是数组的话则遍历数组并调用,然后返回 true。
3 移除事件绑定 removeListener
按 type
取出要删除的监听函数列表 list = event[type]
,当 list
等于要删除的监听函数时, _eventsCount
减一后如果为 0,直接初始化 _events
,否则只删除当前类型的监听函数。
接着往下看,若 typeof list !== 'function'
即 list
为数组时,先确定要删除监听事件的位置 position
,然后删掉对应的函数。
注意:为什么不用 list.splice(postion, 1)
而要专门写一个 spliceOne
来删除呢?
因为这个两参数的方法要比内置的 splice
可能快上 1.5 - 10 倍 !我专门查看了下提交记录,这个版本的方法经过几个开发者改动过最终成为现在这个样子。不得不佩服各路大神对开源的贡献!至于 splice
为什么慢,我没能查到原因,也许需要去看 v8 源码。
4 事件只能执行一次 once
这个方法比较简单,主要能理解 onceWrapper
方法就行。
其它还有一些方法,我不再多写了,基本上原理就是这样。有兴趣的同学可以自己点击前文的源码链接查看。
以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,也希望大家多多支持 码农网
猜你喜欢:- 探索 YOLO v3 实现 - 训练 1
- 探索HyperLogLog算法(含Java实现)
- App 黑白化实现探索,有一行代码实现的方案吗?
- 探索iOS中Block的实现原理
- 探索纯前端实现实时的视频帧预览
- 微热山丘,探索 IoC、AOP 实现原理(二) AOP 实现原理
本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。