关于eventListener的那些事
If multiple identical
EventListeners
are registered on the sameEventTarget
with the same parameters, the duplicate instances are discarded. They do not cause theEventListener
to be called twice, and they do not need to be removed manually with theremoveEventListener()
method. ^1
如果在同一个EventTarget
上添加同一个(reference相同)EventListeners
,该listener只会执行一次,无需手动调用removeEventListener
移除多余的listener
所以在addEventListener
时应尽量避免使用匿名函数的形式,而应该引用已定义好的函数,从而避免之后重复添加listener时被重复执行多次,并且也为removeListener
提供可能性
getEventListeners
有时候我们会想知道document里的全部或某个object上面都绑定了什么listener,除了自己写一段代码^2去查看之外,chrome dev tools也为我们提供了一些方便的接口^3
查看document中的所有listener
使用chrome查看单个object上的listener
直接在chrome的console中使用getEventListeners
接口即可1
getEventListeners(document)
如果想知道listener的函数源码,可以对该listener右键选择 store as global variable, chrome会将其储存为名为tempX
的变量(X为正整数),并打印出该变量内容,即可看到listener函数的具体代码
addEventListener
大多数时候我们在addEventListener
时,都会省略第三个参数options
。要了解options
中的各项参数有什么用途,首先我们需要了解,事件冒泡(Event Bubbling)和事件捕捉(Event Capturing)的执行顺序^4。
事件冒泡 事件捕捉
当触发一个event时,现代浏览器可以有两种执行模式/顺序,默认为事件冒泡
- 事件冒泡,由下至上,child 向 ancestor 依次查找eventListner并执行
- 事件捕捉,由顶至下,由最外层 ancestor 向 child 依次查找、执行, (until it reaches the element that was actually clicked on)
options
^5
options
包括以下几个选项:
capture
,如果为true
,该event会最先dispatch。 before any other children doms’ same event 如同上一节所提到的,如果设置为true
,浏览器会按照事件捕捉的顺序来执行相关listeneronce
,如果为true
,listener最多只会执行一次,之后会自动移除passive
,如果为true
,不会调用preventDefault()
,如果该方法被调用,则该方法会被忽略且抛出警告⚠️。1
Unable to preventDefault inside passive event listener invocation.
passive
为false
时,touch
event会阻塞浏览器的主线程,从而影响scroll的性能,因此一些浏览器(chrome, firefox)将touchstart
touchemove
事件在window
document
document.body
级别的passive
默认值设为true
^6
MDN的这个例子非常好的解释了各种option搭配下的各种情况
另外需要注意的是,我们经常看到这样的写法:1
element.addEventListener('click', myClickHandler, false);
第三个参数并非一个object
,而是一个boolean
,这样的写法是为了兼容一些旧的浏览器,当使用这样的写法时,第三个参数false
指代的是capture
参数
The options argument sets listener-specific options. For compatibility this can be a boolean, in which case the method behaves exactly as if the value was specified as options’s capture. ^7
IE
IE9之前,没有addEventListener
方法,需要使用attachEvent
方法代替,该方法不接受第三个参数 ^8
removeListener
While
addEventListener()
will let you add the same listener more than once for the same type if the options are different, the only optionremoveEventListener()
checks is the capture/useCapture flag. Its value must match forremoveEventListener()
to match, but the other values don’t. ^9
这个就是说,如果在addEventListener
是添加了某些options
,那么移除时也需要添加相应的options
才能成功移除listener 😭 真是麻烦1
2document.addEventListener("click", handler, { passive: true });
document.removeEventListener("click", handler); // fails