DOM4标准的事件监听与滚屏优化

EventTarget.addEventListener()

我们在学习addEventListener()时都只是知道它是用来给事件注册事件处理函数的。但是这种描述并不是很准确,MDN上给我们准确的描述了它的定义。EventTarget.addEventListener()方法将指定的监听器注册到EventTarget上,当该对象触发指定的事件时,指定的回调函数就会被执行。EventTarget目标对象可以是一个文档上的元素ELement、Document、Window或者是任何其它支持事件的对象,例如XMLHTTPRequest
addEventListener()的工作原理是将实现EventListener的函数或者对象添加到调用它的EventTarget上的指定事件类型的事件侦听器列表中。

EventTarget

EventTarget是一个DOM接口,由可以接收事件、并且可以创建侦听器的对象实现。
Element、Document、Window是最常见的EventTargets,但是其它的对象也可以作为EventTargets,比如XMLHTTPRequest、AudioNode、AudioContext等等。
许多EventTargets包括(elements、documents、windows)支持通过onEvent特性和属性设置事件处理函数event handlers
首先EventTarget()是一个构造函数,通过实例化new EventTarget()构造函数创建一个新的EventTarget实例对象,在EventTarget.prototype上存在三个方法:addEventListener()dispatchEvent()removeEventListener()addEventListener()方法在EventTraget上注册特定的事件类型的事件处理函数。removeEventListener()方法删除EventTarget事件处理函数。dispatchEvent()将事件分派到EventTarget
image.png
那么EventTarget事件目标构造函数存在的意义是什么呢?我们看下面的例子中,通过EventTarget构造函数实例化的对象可以继承EventTarget.prototype方法,那么就说明此时的obj对象是一个EventTarget对象。而一个普通对象,因为不是EventTarget对象,所以不能够继承到EventTarget.prototype的方法。

  1. EventTarget构造函数的第一个作用就是创建EventTarget对象。
  2. 让其它的EventTargets对象通过原型链的方式继承到EventTarget.prototype上的方法,例如window、document、Element
const obj = new EventTarget();
obj.addEventListener();const obj = {};
obj.addEventListener(); // Uncaught TypeError: obj.addEventListener is not a function.

为什么window对象可以通过window.addEventListener()的形式注册事件处理函数?window对象种并不存在addEventListener()方法,而addEventListener()方法存在EventTarget.prototype上,但是由于原型链的作用。window对象通过原型链的方式从EventTarget.prototype上继承addEventListener()方法,这也正是window对象为什么能够调用addEventListener()方法的原因。

console.log(window);

image.png

EventTarget的工作方式及简单实现

EventTarget的工作方式主要是利用EventTarget.prototype上的方法。
addEventListener()方法:在EventTarget上注册特定事件类型的事件处理程序。
removeEventListener()方法:在EventTarget中删除事件侦听器事件处理函数。
dispatchEvent()方法:将事件派发到EventTarget上。
熟悉方法之后,我们就要将这几个方法进行重写。
封装EventTarget构造函数的思路是什么呢?首先我们需要分析事件到底是处于什么样的时机触发的呢?我们如何去执行事件触发之后的事件处理函数?其实也是很简单的,我们没有办法具体的控制事件处理函数执行的时机,因为事件处理函数是在事件触发的时候执行的,但是我们不知道事件具体是在什么时候触发的。那么如何处理呢?我们可以通过数组的方式,将该事件类型绑定的事件处理函数都存放到数组内部保存,当事件派发dispatch的时候,将数组中存放的事件处理函数都拿出来执行。

/**
*	EventTarget构造函数
* listeners:存放事件类型,事件回调函数的对象
*/
var EventTarget = function() {this.listeners = {};
}/**
* 为什么在prototype重写声明一遍listeners,因为让其它EventTargets对象继承,
* 例如:window、document
*/
EventTarget.prototype.listeners = null;/**
* @description: addEventListener
* type事件类型不存在listener中,我们就创建一个数组,用来存放该事件回调函数;
* type事件类型存在listener中,将事件回调函数放入对应的事件类型数组中;
* @param {*} type: 事件类型
* @param {*} callback: 事件回调函数
* @return {*} undefined
*/
EventTarget.prototype.addEventListener = function(type, callback) {// 事件类型是否存在listeners中if (!(type in this.listeners)) {this.listeners[type] = [];}this.listeners[type].push(callback);
}/**
* @description: removeEventListener
* 移除绑定的事件处理函数,注意 stack[i] === callback,
* 这就是为什么你需要移除事件监听函数时,必须在addEventListener绑
*	定事件处理函数是具名函数的原因,因为匿名函数无法判断是否相等。
* @param {*} type: 事件类型
* @param {*} callback: 事件回调函数
* @return undefined
*/
EventTarget.prototype.removeEventListener = function(type, callback) {// 事件类型是否存在listeners中if (!(type in this.listeners)) {return;}// stack表示该事件回调数组[]var stack = this.listeners[type],len = stack.length;for (var i = 0; i < len; i++) {if (stack[i] === callback) {this.listeners.splice(i, 1);}	}
}/**
* @description: dispatchEvent
* 向一个指定的事件目标派发一个事件,
* 并以合适的顺序同步调用目标元素相关的
*	事件处理函数。
* @param {*} event:要派发的事件对象
* @param {*} target:用来初始化事件和决定将会触发目标
*/
EventTarget.prototype.dispatchEvent = function(event) {// 事件类型是否存在listeners中if (!(event.type in this.listeners)) {return;}var stack = this.listeners[event.type],len = stack.length;event.target = this;for (var i = 0; i < len; i++) {stack[i].call(this, event);}
}

EventTarget.dispatchEvent()深入到Event构造函数

EventTarget.dispatchEvent()方法与浏览器原生事件有什么不同?浏览器原生事件,是由DOM派发的,并通过Event loop异步调用事件处理程序,而dispatchEvent()则是同步调用事件处理程序。在调用dispatchEvent()后,所有监听该事件的事件处理程序将在代码前执行返回。
dispatchEvent()方法是create-init-dispatch过程中的最后一步,用于将事件调用到实现的事件模型中。可以利用Event构造函数创建事件。这是MDN文档上对于dispatchEvent方法的介绍,既然介绍到了Event构造函数,我们就一起来看看如何自定义事件对象Event
注意一哈,事件是否能够取消,事件处理函数中是否阻止过事件默认行为,这都是可以获取到的。e.cancelable作为Event实例的只读属性,表明事件是否可以被取消。e.defaultPrevented判断处理函数中是否阻止过事件默认行为,换句话说就是在事件处理函数中是否调用过e.preventDefault()方法。

Event()构造函数,创建一个新的事件对象Event。event = new Event(typeArg, eventInit);typeArg: 表示所创建事件的名称。eventInit:是EventInit类型的字典,接受以下的字段:·"bubbles",可选,Boolean类型,默认值为false,表示事件是否冒泡。·"cancelable",可选,Boolean类型,默认值为false,表示该事件是否能被取消。·"composed",可选,Boolean类型,默认值为false,指示事件是否会在影子DOM根节点之外触发侦听器。

熟悉Event构造函数,我们来尝试自定义一个事件,然后利用dispatchEvent()方法将事件派发到EventTarget对象上。下面例子中,我自定义了一个事件see,并且这个see事件支持冒泡,不支持取消,通过dispatchEvent方法将事件派发到EventTarget对象(oDiv)上,此时oDiv元素就能够监听到我自定义的see事件。
注意下面例子中,虽然事件see的事件处理程序中调用了e.preventDefault()方法,但是e.defaultPrevented依旧返回false,这是为什么?因为see事件在定义的时候,我们将cancelable字段设置为false,也就是表明事件see不可取消,事件处理程序中无法监听回调中停止事件。所以,e.defaultPrevented字段的结果返回false

var ev = new Event('see', {bubbles: true,cancelable: false
});var oDiv = document.getElementsByTagName('div')[0];oDiv.addEventListener('see', function(e) {e.preventDefault();console.log(e.defaultPrevented); // falseconsole.log('Listening event see....');console.log(e.cancelable); // false 事件不可取消
});oDiv.dispatchEvent(ev);

EventTarget.addEventListener()深入到滚屏优化

MDN文档上指出EventTarget.addEventListener()方法将指定的监听器注册到EventTarget上,当该对象触发指定的事件时,指定的回调函数就会被执行。事件目标可以是一个文档上的元素Element、Document、Window或者任何支持事件的对象,比如XMLHTTPRequest
addEventListener()工作原理是将实现EventListener的函数或对象添加到调用它的EventTarget上指定事件类型的事件侦听器列表中,与我们上面重写addEventListener()方法的逻辑一致。
上面简述是MDN文档对addEventListener()方法的定义,我们之前学习addEventListener()方法时并没有仔细的看addEventListener()方法的参数,Vue中的事件修饰符与addEventListener()方法中的options参数特别相似。
addEventListener(eventType, handler, useCapture || options);这是addEventListener()方法标准的语法,其中eventType表示监听的事件类型,hanlder表示事件处理函数,useCapture表示事件流中两种事件传播方式,false表示选择事件冒泡的方式触发事件处理函数,true表示选择事件捕获的方式触发事件处理函数。

oDiv.addEventListener('click',function(){},false);

上面的例子中,是我们最常用的方式。但是在DOM4的标准里,addEventListener()方法中的还可以设置options参数,options参数表示:一个指定有关listener(事件处理程序) 属性的可选参数对象。可选的参数默认都是false,可选参数有:
captureBoolean,表示listener会在该类型的事件捕获阶段阶段传播到该EventTarget时触发,与我们上面分析的useCapture是同一个意思。
onceBoolean,表示listener在添加之后最多调用一次。如果是true,listener会在其被调用之后自动移除。
passiveBoolean,设置为true时,表示listener永远不会调用ev.preventDefault()方法,如果你仍然在listener调用了ev.preventDefault()方法,浏览器会在控制台中抛出警告unable to prevetDefault inside passive event listener invocation;(无法在被动事件侦听器调用内预先设置默认值)。

滚屏优化

在学习addEventListener()方法中的passive字段时,MDN文档中上提出了一个“使用passive改善滚屏性能”的概念。
:::info
下面例子中是在Chrome浏览器下中测试的结果。
:::
在了解“使用passive改善的滚屏性能”概念之前,我们先看下面的例子。我们在window对象上增加touchstart事件的事件处理函数function(e){}。只要我们触摸到屏幕开始滚动的时候,就会执行调用绑定的事件处理函数function(e){}

window.addEventListener('touchstart',function(e){console.log('Listening scroll....');
});

那么我现在想阻止touchstart事件的默认行为,touchstart事件的默认行为是什么呢?touchstart事件的默认行为其实就是srcoll滚动。因为你触摸点击屏幕开始滚动的时候,touchstart事件的listener侦听器就开始执行。那我现在尝试阻止touchstart事件的默认行为,例如下面的例子。
我在事件处理函数listeners中增加了e.preventDefault()语句,希望阻止touchstart事件的默认行为,也就是说我不想让屏幕滚动了,并且我还在listeners中增加e.defaultPrevented语句判断是否调用过e.preventDefault()方法。但是结果如下面的截图一样,这样的方式不仅仅没有成功的阻止touchstart事件的默认行为,而且浏览器还抛出了异常。这是为什么?
我们通过对异常的分析,错误提示我们:passive无法阻止去调用默认行为,事件处理函数listener取决于目标开始时的passive状态。换句话来说,就是passive字段现在的状态相当于设置成true,因为上面我们说过,passivetrue的时候,listener永远不会调用ev.preventDefault(),如果你强制调用,则会抛出异常。

window.addEventListener('touchstart',function(e){e.preventDefault();console.log('Listening scroll....');console.log(e.defaultPrevented); // false
});

image.png
为什么会出现上述的这种情况呢?
根据MDN文档,passive选项的默认值始终为false。但是某些浏览器(特别是Chrome和Firefox)已将文档级节点Window、Document、Document.bodytouchstarttouchmove事件的passive选项默认值设置为true。所以上面两种情况在listeners中调用ev.preventDefault()方法并不能够成功阻止事件的默认行为。但是浏览器为什么要这样做?
因为在监听touchstart事件的时候,用户如果触发了touchstart事件,那么touchstart事件绑定的处理函数listeners在执行的时候,内部如果有阻止默认行为的代码时,就不会再去执行默认行为了。如果内部没有阻止默认行为的代码时,下一步就会执行默认行为。但是无论是否阻止默认行为,它都会有一个等待的时间,因为必须等待listener执行程序完成后,再去执行默认行为(滚动),此时等待的时间会造成滚动的卡顿。所以执行的顺序是:listeners —> 执行默认行为,所以这种执行顺序在主线程中内部存在非常大的性能问题,由于有等待的时间,导致滚动卡顿。所以有些浏览器针对这种问题,将passive默认值改为true
那么将passive设置为true有什么好处?passive设置为true时,程序会开启两个线程进行处理滚动的问题,一个线程是处理listeners的执行,一个线程是处理执行默认行为,所以正是因为这个原因,MDN文档上指出“passive优化改善滚屏性能”的概念。
下面的例子,成功的取消了touchstart事件的默认行为。注意首先我们说以上测试的结果都是在chrome浏览器中,其次我们需要掌握的不是如何取消touchstart的默认行为,而是passive为什么能够改善滚屏性能的原因?最后要了解addEventListener()方法中的options参数。

window.addEventListener('touchstart',function(e){e.preventDefault();console.log('Listening scroll....');console.log(e.defaultPrevented); // false
},{passive:false
});

image.png

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://xiahunao.cn/news/2979399.html

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

字符长、看不懂、费率飙升|Runes协议上线后发生了什么?

作者&#xff1a;比特里里 X/推&#xff1a;lilyanna_btc 1、字符数长了&#xff0c;单词都完整了&#xff0c;反而看不懂了 由于 Runes 协议的字符长度限制&#xff0c;大部分的票都在 13 个字符及以上&#xff0c;人名、域名、slogan&#xff0c;各类玩法都出来了。很多人适…

文本向量化模型新突破——acge_text_embedding勇夺C-MTEB榜首

在人工智能的浪潮中&#xff0c;以GPT4、Claude3、Llama 3等大型语言模型&#xff08;LLM&#xff09;无疑是最引人注目的潮头。这些模型通过在海量数据上的预训练&#xff0c;学习到了丰富的语言知识和模式&#xff0c;展现了出惊人的能力。在支撑这些大型语言模型应用落地方面…

JS----前端不同格式的 UUID生成

前端 UUID生成 export const generateUUID () > {let timestamp new Date().getTime()let performanceNow (typeof performance ! undefined && performance.now && performance.now() * 1000) || 0return xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx.replac…

免费好用的AI绘画生成器推荐,建议收藏!

AI绘画生成器可以根据设计师的需求生成各种风格、主题的绘画作品&#xff0c;无论是现代抽象&#xff0c;还是古典风景&#xff0c;都能应有尽有。任何人只需几步简单操作&#xff0c;都可以轻松使用这款工具&#xff0c;就能生成出令人惊艳的作品。那么免费好用的AI绘画生成器…

docker部署通义千问-7B-Chat的openai-api环境

服务器环境&#xff1a; 显卡驱动&#xff1a;Driver Version: 530.30.02 CUDA版本&#xff1a;CUDA Version: 12.1 显卡&#xff1a;NVIDIA GeForce RTX 3090共4张 注意&#xff1a;最好把显卡驱动升级到530&#xff0c;CUDA版本之前使用11.7有问题。 一、下载模型文件 …

Linux之安装Nginx

目录 传送门前言一、快速安装二、反向代理语法1、基本语法2、location语法1. 基本语法2. 匹配规则3. 修饰符4. 权重5. 嵌套location6. 其他指令7.案例 三、配置反向代理 传送门 SpringMVC的源码解析&#xff08;精品&#xff09; Spring6的源码解析&#xff08;精品&#xff0…

2024年深圳杯东三省数学建模联赛A题论文首发第二种思路

深圳杯A题论文代码分享资料链接&#xff1a;链接&#xff1a;https://pan.baidu.com/s/1L2NVgoefSW-yuqZjEB3wcw 提取码&#xff1a;sxjm 问题一 数据转换&#xff1a; 首先&#xff0c;我们将监测站的经纬度坐标转换为基于米的笛卡尔坐标系。这是因为在地面上的大尺度距离…

selenium‘拟人包装‘设置

1、设置header,proxy 1.1关于user-agent 输入about:version 找到user-agent: import requests # 引用requests库 from selenium import webdriver#载入浏览器驱动#header&#xff0c;proxy设置 optionswebdriver.ChromeOptions()#实例化浏览器参数设置options.add_argument…

探索深度与广度的平衡:迭代加深深度优先搜索技术解析

探索深度与广度的平衡&#xff1a;迭代加深深度优先搜索技术解析 迭代加深深度优先搜索&#xff08;IDDFS&#xff09;的基本原理伪代码C语言实现讨论结论 迭代加深&#xff08;Iterative Deepening Depth-First Search, IDDFS&#xff09;是一种用于解决搜索问题的方法&#x…

解决配置Tomcat时,找不到war和war exploded问题

解决配置Tomcat时&#xff0c;找不到war和war exploded问题 文章目录 解决配置Tomcat时&#xff0c;找不到war和war exploded问题前言一、解决方法&#xff1a;1. war exploded2. war 总结 前言 提示&#xff1a;以下是本篇文章正文内容&#xff1a; 一、解决方法&#xff1a;…

spring的跨域问题

跨域问题 什么是跨域解决跨域 什么是跨域 跨域问题本质是浏览器的一种保护机制&#xff0c;它的初衷是为了保证用户的安全&#xff0c;防止恶意网站窃取数据。如果出现了以下情况中的任意一种&#xff0c;那么它就是跨域请求&#xff1a; 1、协议不同&#xff0c;如 http 和 h…

网站想实现HTTPS访问需要有哪些步骤?

网站要实现HTTPS访问&#xff0c;以确保数据传输安全和提升用户信任度&#xff0c;主要需按以下步骤操作&#xff1a; 1. 购买或申请SSL证书&#xff1a; - 根据网站类型和需求&#xff0c;选择合适的SSL证书&#xff1a;DV&#xff08;域名验证&#xff09;、OV&#xff08;组…

翻页电子图书制作小技巧分享给你

当今社会&#xff0c;二维码已经成为了信息传递的重要方式之一&#xff0c;其在电子商务、广告营销、活动推广等领域广泛应用。而如何将二维码巧妙地融入电子画册中&#xff0c;制作出高端、具有吸引力的作品&#xff0c;成为了许多设计师和营销人员关注的焦点 但是很多人却不知…

TCP协议核心一文搞懂<随手笔记>

1.简介 传输控制协议&#xff0c;是一种面向连接的、可靠的、基于IP的传输层协议。 TCP工作于传输层&#xff0c;IP在网络层&#xff0c;ARP在数据链路层 源端口和目的端口&#xff1a; 各占两个字节&#xff0c;这两个值加上IP首部中的源端IP地址和目的端IP地址唯一确定一个T…

华为sr-mpls policy配置案例

SR&#xff0d;MPLS POLICY在ensp上面做不了&#xff0c;这是官方上的配置

解压RPM包---rpm2cpiocpio的使用

RPM包括是使用cpio格式打包的&#xff0c;因此可以先转成cpio然后解压&#xff0c;如下所示&#xff1a; rpm2cpio xxx.rpm | cpio -div

【C++】项目级的组织结构与Cmake编译

文章目录 C项目级的组织结构与Cmake编译分文件编写程序C项目级的组织结构Cmake编译 C项目级的组织结构与Cmake编译 分文件编写程序 (1) 创建后缀名为.h的头文件max.h&#xff0c;并在其中写函数的声明 #include<iostream> using namespace std; int max(int a, int b)…

【Linux】文件目录及路径表示

1. Linux目录结构 在 Linux 系统中&#xff0c;有几个目录是比较重要的&#xff0c;平时需要注意不要误删除或者随意更改内部文件。 /etc&#xff1a; 这个是系统中的配置文件&#xff0c;如果更改了该目录下的某个文件可能会导致系统不能启动。 /bin, /sbin, /usr/bin, /usr…

前端JS必用工具【js-tool-big-box】,获取浏览器参数、cookie、localStorage的存取

这一小节&#xff0c;我们针对js-tool-big-box工具做一些使用讲解&#xff0c;主要获取浏览器参数、cookie、localStorage的存取方面的。 这些方法差不多每次项目中要么用不到&#xff0c;要么就自己写一份&#xff0c;轮子造的很重复啊&#xff0c;而且localStorage有时候要求…

Docker Compose 的安装和使用详解

Docker Compose 是 Docker 官方开源的容器编排(Orchestration)项目之一,用于快速部署分布式应用。本文将介绍 Docker Compose 的基本概念、安装流程及使用方法。 简介 Compose 项目是 Docker 官方的开源项目,负责实现对 Docker 容器集群的快速编排。从功能上看,Docker C…