关于eventListener的那些事

If multiple identical EventListeners are registered on the same EventTarget with the same parameters, the duplicate instances are discarded. They do not cause the EventListener to be called twice, and they do not need to be removed manually with the removeEventListener() 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

事件冒泡 事件捕捉

img

当触发一个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,浏览器会按照事件捕捉的顺序来执行相关listener
  • once,如果为true,listener最多只会执行一次,之后会自动移除
  • passive,如果为true,不会调用preventDefault(),如果该方法被调用,则该方法会被忽略且抛出警告⚠️。
    1
    Unable to preventDefault inside passive event listener invocation.

passivefalse时,touchevent会阻塞浏览器的主线程,从而影响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 option removeEventListener() checks is the capture/useCapture flag. Its value must match for removeEventListener() to match, but the other values don’t. ^9

这个就是说,如果在addEventListener是添加了某些options,那么移除时也需要添加相应的options才能成功移除listener 😭 真是麻烦

1
2
document.addEventListener("click", handler, { passive: true });
document.removeEventListener("click", handler); // fails