探索JavaScript事件流:DOM中的神奇旅程
🎬 江城开朗的豌豆:个人主页
🔥 个人专栏 :《 VUE 》 《 javaScript 》
⛺️ 生活的理想,就是为了理想的生活 !
目录
引言
1. 事件流的发展流程
1.1 传统的DOM0级事件
1.2 DOM2级事件和addEventListener方法
1.3 W3C DOM3级事件
1.4 React与Virtual DOM
2. 事件流的属性
2.1 事件捕获阶段
2.2 目标阶段
2.3 事件冒泡阶段
2.4 事件对象
3. 事件流的应用场景
3.1 事件处理
3.2 事件代理
3.3 事件委托
4. 示例代码
引言
JavaScript中的事件流是一种机制,用于描述和处理事件在DOM树中的传播过程。了解事件流的属性和工作原理对于编写高效的事件处理代码和实现复杂的交互功能至关重要。本文将详细介绍JavaScript事件流的发展流程、属性以及应用场景,并提供一些代码示例和引用资料,帮助读者深入理解并应用这一重要的前端技术。
1. 事件流的发展流程
事件流在前端的发展过程中经历了一些变化和演进。下面简要介绍了事件流的发展历程:
1.1 传统的DOM0级事件
在早期的JavaScript中,事件处理是通过在DOM元素上直接定义事件处理属性来实现的,称为DOM0级事件。例如,可以通过为按钮元素的
onclick
属性赋值一个函数来定义点击事件的处理程序。
const button = document.getElementById('myButton');
button.onclick = function() {console.log('按钮被点击');
};
这种方式简单直接,但是有一个缺点是无法同时为一个元素的同一个事件类型添加多个处理程序。
1.2 DOM2级事件和addEventListener方法
随着DOM2级事件的引入,新增了
addEventListener
方法,提供了更强大和灵活的事件处理能力。addEventListener
方法允许为一个元素的同一个事件类型添加多个处理程序,并且可以控制事件的捕获阶段。
const button = document.getElementById('myButton');
button.addEventListener('click', function() {console.log('按钮被点击');
});
通过
addEventListener
方法,可以在一个元素上同时添加多个处理程序,而且可以使用removeEventListener
方法移除指定的处理程序。
1.3 W3C DOM3级事件
W3C DOM3级事件进一步扩展了事件的类型和属性,引入了更多的事件类型和特性,以满足不断增长的前端开发需求。DOM3级事件规范定义了新的事件类型,如滚动事件、触摸事件、过渡事件等,以及一些新的事件属性和方法,提供更丰富的事件处理能力。
const element = document.getElementById('myElement');
element.addEventListener('scroll', function(event) {console.log('元素滚动事件');
});
DOM3级事件的引入丰富了事件处理的能力,使得开发者可以更灵活地响应各种类型的事件。
1.4 React与Virtual DOM
随着React等前端框架的出现,事件处理机制也发生了一些变化。React通过Virtual DOM的概念,将事件处理从直接操作DOM转移到组件层面进行管理。React利用了合成事件(
SyntheticEvent)来处理事件,实现了跨浏览器的一致性和性能优化。
在React中,事件处理程序是通过特定的语法和属性绑定到组件的,而不是直接操作DOM元素。
class MyComponent extends React.Component {handleClick() {console.log('按钮被点击');}render() {return <button onClick={this.handleClick}>点击按钮</button>;}
}
通过使用合成事件,React能够更高效地管理事件处理,并提供了更好的性能和开发体验。
2. 事件流的属性
事件流涉及到三个主要的概念:事件捕获阶段、目标阶段和事件冒泡阶段。了解这些阶段和相关的属性对于理解事件流的机制至关重要。
2.1 事件捕获阶段
事件捕获阶段是事件流的第一个阶段,从根节点开始向下传播到目标元素。在事件捕获阶段中,事件依次经过每个父元素,直到达到目标元素。
在事件捕获阶段,可以使用
addEventListener
的第三个参数指定事件处理程序在捕获阶段中执行。
element.addEventListener('click', handler, true);
2.2 目标阶段
目标阶段是事件流的第二个阶段,事件到达目标元素后被触发执行事件处理程序。
2.3 事件冒泡阶段
事件冒泡阶段是事件流的最后一个阶段,事件从目标元素开始向上冒泡,依次经过每个父元素,直到达到根节点。
在事件冒泡阶段,可以使用
addEventListener
的第三个参数设置为false
或省略来指定事件处理程序在冒泡阶段中执行(默认值)。
element.addEventListener('click', handler, false);
// 或
element.addEventListener('click', handler);
2.4 事件对象
在事件处理程序中,可以通过事件对象访问和操作相关的事件信息。事件对象提供了一些属性和方法,可以获取事件的类型、目标元素、鼠标坐标等信息。
例如,可以通过事件对象的
type
属性获取事件类型:
element.addEventListener('click', function(event) {console.log(event.type); // 输出 'click'
});
3. 事件流的应用场景
事件流在前端开发中具有广泛的应用场景,下面介绍几个常见的应用场景:
3.1 事件处理
事件流提供了一种机制,用于处理和响应用户的交互操作。通过在目标元素上注册事件处理程序,可以捕获和处理用户触发的事件,实现交互功能。
例如,通过在按钮上注册click
事件处理程序,可以
在按钮被点击时执行相应的代码逻辑。
const button = document.getElementById('myButton');
button.addEventListener('click', function(event) {console.log('按钮被点击');
});
3.2 事件代理
事件代理(Event Delegation)是一种常见的优化技术,用于处理大量具有相似行为的子元素事件。通过在父元素上注册事件处理程序,可以利用事件冒泡机制,统一管理子元素的事件处理。
例如,可以在父元素上注册
click
事件处理程序,根据触发事件的具体子元素进行不同的操作。
const list = document.getElementById('myList');
list.addEventListener('click', function(event) {if (event.target.tagName === 'LI') {console.log('项目被点击');}
});
3.3 事件委托
事件委托是一种通过将事件处理委托给父元素来提高性能和简化代码的技术。它利用事件冒泡机制,在父元素上注册一个事件处理程序,处理多个子元素的相同事件。
例如,可以在父元素上注册click
事件处理程序,根据触发事件的子元素的不同类别执行不同的操作。
const container = document.getElementById('myContainer');
container.addEventListener('click', function(event) {if (event.target.classList.contains('button')) {console.log('按钮被点击');} else if (event.target.classList.contains('link')) {console.log('链接被点击');}
});
4. 示例代码
下面是一些示例代码,演示了事件流的应用和相关的属性:
<button id="myButton">点击按钮</button>
<ul id="myList"><li>项目1</li><li>项目2</li><li>项目3</li>
</ul>
<div id="myContainer"><button class="button">按钮</button><a href="#" class="link">链接</a>
</div><script>// 事件处理示例const button = document.getElementById('myButton');button.addEventListener('click', function(event) {console.log('按钮被点击');});// 事件代理示例const list = document.getElementById('myList');list.addEventListener('click', function(event) {if (event.target.tagName === 'LI') {console.log('项目被点击');}});// 事件委托示例const container = document.getElementById('myContainer');container.addEventListener('click', function(event) {if (event.target.classList.contains('button')) {console.log('按钮被点击');} else if (event.target.classList.contains('link')) {console.log('链接被点击');}});
</script>
请大家不吝赐教,在下方评论或者私信我,十分感谢🙏🙏🙏.
✅ 认为我某个部分的设计过于繁琐,有更加简单或者更高逼格的封装方式
✅ 认为我部分代码过于老旧,可以提供新的API或最新语法
✅ 对于文章中部分内容不理解
✅ 解答我文章中一些疑问
✅ 认为某些交互,功能需要优化,发现BUG
✅ 想要添加新功能,对于整体的设计,外观有更好的建议
最后感谢各位的耐心观看,既然都到这了,点个 👍赞再走吧
相关文章:

探索JavaScript事件流:DOM中的神奇旅程
🎬 江城开朗的豌豆:个人主页 🔥 个人专栏 :《 VUE 》 《 javaScript 》 ⛺️ 生活的理想,就是为了理想的生活 ! 目录 引言 1. 事件流的发展流程 1.1 传统的DOM0级事件 1.2 DOM2级事件和addEventListener方法 1.3 W3C DOM3级…...

听GPT 讲Rust源代码--library/std(8)
题图来自Why is Rust programming language so popular?[1] File: rust/library/std/src/sys/sgx/abi/reloc.rs 在Rust源代码中,sgx/abi/reloc.rs文件的作用是定义了针对Intel Software Guard Extensions (SGX)的重定位相关结构和函数。 该文件中的Rela 结构定义了…...

Hbase基本使用,读写原理,性能优化学习
文章目录 HBase简介HBase定义HBase数据模型**HBase** **逻辑结构****HBase** **物理存储结构****HBase** **基本架构** HBase 入门**HBase** **安装部署****HBase** 配置文件**HBase** 启动停止**HBase** **访问页面****HBase** **高可用****HBase Shell****HBase API**HBaseCo…...

添加主仓库后报错error: remote upstream already exists.
可能的原因 远程名 upstream 已经被使用: 这通常意味着你在之前已经添加了一个名为 upstream 的远程仓库。 解决方案 检查现有的远程仓库: 运行 git remote -v 来查看所有配置的远程仓库。这个命令会列出所有远程仓库的URL,你可以检查是否已…...

香港服务器如何做负载均衡?
在现代互联网时代,随着网站访问量的不断增加,服务器的负载也越来越重。为了提高网站的性能和可用性,负载均衡成为了一种常见的解决方案。 什么是负载均衡? 负载均衡是一种技术解决方案,用于在多个服务器之间分配负载&#…...

前端 :用HTML , CSS ,JS 做一个秒表
1.HTML: <body><div id "content"><div id "top"><div id"time">00:00:000</div></div><div id "bottom"><div id "btn_start">开始</div><div …...

BIOS MBR UEFI GPT详解
先来看下名词 启动方式: BIOS:Basic Input Output System,中文名称"基本输入输出系统"。 UEFI:Unified Extensible Firmware Interface,中文名称"统一的可扩展固件接口"。 Legacy:…...

2023NOIP A层联测20-点餐
一家新的餐馆开业了,为了吸引更多的顾客,每样餐品都有打折的活动。特别的,餐馆内一共有𝑛样菜品,编号从 1 1 1 到 n n n,每样菜品每人最多只能点一次。对于第 i i i 种菜品,其包含两种价格&a…...

3D LUT 滤镜 shader 源码分析
最近在做滤镜相关的渲染学习,目前大部分 LUT 滤镜代码实现都是参考由 GPUImage 提供的 LookupFilter 的逻辑,整个代码实现不多。参考网上的博文也有各种解释,参考了大量博文之后终于理解了,所以自己重新整理了一份,方便…...

五分钟理解Java跨平台原理(适合小白)
JVM通俗的理解 Java语言的一个非常重要的特点就是与平台的无关性。而使用Java虚拟机,即JVM(Java Virtual Machine)是实现这一特点的关键。JVM是一种用于计算设备的规范,它是一个虚构出来的计算机,是通过在实际的计算机…...

从初级测试工程师到测试专家,你的晋升路线是什么?
最近,我们讨论了软件测试工程的的分级,大家都贡献了自己的想法。 对于大家来说,软件测试人的分级其实也代表了我们的进阶方向,职业发展。总体来说,测试工程师未来发展有三个方向: 技术精英 行业专家 管理…...

合肥中科深谷嵌入式项目实战——人工智能与机械臂(四)
订阅:新手可以订阅我的其他专栏。免费阶段订阅量1000 python项目实战 Python编程基础教程系列(零基础小白搬砖逆袭) 作者:爱吃饼干的小白鼠。Python领域优质创作者,2022年度博客新星top100入围,荣获多家平台专家称号。…...

Zynq-Linux移植学习笔记之64- 国产ZYNQ在linux下配置国产5396芯片
1、背景介绍 复旦微ZYNQ通过SPI配置国产JEM5396,框图如下: 现在需要在linux下的应用程序内配置JEM5396的寄存器。其中FMQL和进口的XILINX ZYNQ类似,JEM5396和进口的BCM5396兼容。因此可以参考进口ZYNQ在linux下配置BCM5396过程。Zynq-Linux移…...

系统架构设计师-第19章-大数据架构设计理论与实践-软考学习笔记
传统数据处理系统存在的问题 传统数据处理系统存在以下问题: 1. 数据孤岛问题:不同部门或系统之间的数据隔离,数据无法共享和整合。 2. 数据不一致性问题:由于数据维护分散,同一数据在不同系统或部门中可能存在不同…...

论坛搭建.
目录 一.配置软件仓库 二.安装http php miriadb 三.配置数据库 四.源码拖拽并解压 五.防火墙通过 六.浏览器安装测试 七.界面参数设置 一.配置软件仓库 1.进入仓库目录 cd /etc/yum.repos.d 2.创建仓库文件 vim local.repo 3.在 local.repo中写入:(粘贴的时候注意位…...

三种前端埋点方式
什么是埋点 埋点是数据采集领域(尤其是用户行为数据采集领域)的术语,指的是针对特定用户行为或事件进行捕获、处理和发送的相关技术及其实施过程。比如用户某个icon点击次数、观看某个视频的时长等等。 我们可以知道埋点实际上是对特定事件或…...

html获取网络数据,列表展示 第二种
html获取网络数据,列表展示 第二种 js遍历json数组中的json对象 image.png || - 判断数据是否为空,为空就显示 - <!DOCTYPE html> <html><head><meta charset"utf-8"><title>网页列表</title><script …...

【Python 算法】信号处理通过陷波滤波器准确去除工频干扰
对于一个信号来说通常汇入工频噪声往往是因为交流电产生的电泳,影响了我们信号采集导致信号上存在工频干扰。 那么matlab去除工频干扰可以通过陷波滤波器实现。 通常使用scipy.signal实现信号的处理。 Scipy的信号处理模块(scipy.signal)来创…...

Redis(08)| 线程模型
一、redis 的线程模型 redis 内部使用文件事件处理器 file event handler,它是单线程的,所以redis才叫做单线程模型。它采用IO多路复用机制同时监听多个 socket,将产生事件的 socket 压入内存队列中,事件分派器根据 socket 上的事…...

Java14-16新特性
目录 一、Java14新特性 1、instanceof模式匹配 2、友好的空指针(NullPointerException)提示 3、record类型 二、Java15新特性 1、Sealed Classes 2、CharSequence新增方法 3、TreeMap新增方法 4、文本块 5、无需配置环境变量 三、Java16新特性 1、包装类构造方法的…...

中兴再推爆款,双2.5G网口的巡天AX3000Pro+仅需299元
10月30日消息,中兴新款路由器中兴巡天AX3000Pro将于10月31日20:00正式开售,当前可在天猫、京东及红魔商城进行预约,首发价格299元。 据了解,中兴巡天AX3000Pro是中兴智慧家庭推出的巡天系列新品,也是当前市场上唯一一款300元价位内配备双2.5G网口的路由器。 中兴巡天AX3000Pro…...

【系统架构】架构风格专题
目录 1、定义 2、通用架构风格分类 3、架构风格比较 4、示例:管道-过滤 VS 数据仓库)比较因素分析 1、定义 架构风格:描述某一特定应用领域中系统组织方式的惯用模式,反映了领域中众多系统所共有的结构和语义特性,…...

【Qt】盒子布局、网格布局、表单布局和堆栈布局
盒子布局 QBoxLayout可以在水平方向或垂直方向上排列控件,分别派生了QHBoxLayout、QVBoxLayout子类。 QHBoxLayout:水平布局,在水平方向上排列控件,即:左右排列。QVBoxLayout:垂直布局,在垂直…...

GO语言,半自动打怪
仅供学习参考,切勿用于商业用途 package mainimport ("fmt""github.com/go-vgo/robotgo""math/rand""time" )const (taskNum 7 )type Task struct {Name stringSleepTime1 intSleepTime2 intFunc func() }fu…...

【Java 进阶篇】Java登录案例详解
登录是Web应用程序中常见的功能,它允许用户提供凭证(通常是用户名和密码)以验证其身份。本文将详细介绍如何使用Java创建一个简单的登录功能,并解释登录的工作原理。我们将覆盖以下内容: 登录的基本概念创建一个简单的…...

Vue 菜单导航栏,轮播图
导航菜单栏结构和样式代码实现 一级导航栏 views/HomeView.vue <template><div><Shortcut></Shortcut><Header></Header><div class"inner"><Navigation></Navigation></div><div>我是主页&l…...

讲述为什么要学习Adobe XD以及 Adobe XD下载安装
首先 我们要了解 Adobe XD 是个什么东西 XD是Adobe公司专门开发出来面向交互、界面设计的矢量绘图工具。 然后是 他可以做什么? 最基本的 可以做UI界面设置 所有 手机 平板 电脑等设备的UI界面 我们都可以通过XD完成 还有就是原型设置 我们可以做各种界面图 还有…...

Netty复习:(1)Http server: hello world
一、加依赖 <dependency><groupId>io.netty</groupId><artifactId>netty-all</artifactId><version>4.1.72.Final</version></dependency>二、创建自定义的handler package cn.edu.tju.handler;import io.netty.buffer.ByteB…...

【Python 千题 —— 基础篇】加法计算
题目描述 题目描述 编写一个程序,接受用户输入的两个数字,然后计算这两个数字的和,并输出结果。 输入描述 输入两个数字,用回车隔开两个数字。 输出描述 程序将计算这两个数字的和,并输出结果。 示例 示例 ① …...

基于纵横交叉算法的无人机航迹规划-附代码
基于纵横交叉算法的无人机航迹规划 文章目录 基于纵横交叉算法的无人机航迹规划1.纵横交叉搜索算法2.无人机飞行环境建模3.无人机航迹规划建模4.实验结果4.1地图创建4.2 航迹规划 5.参考文献6.Matlab代码 摘要:本文主要介绍利用纵横交叉算法来优化无人机航迹规划。 …...