JS 事件流机制详解:冒泡、捕获与完整事件流

news2025/6/9 17:33:31

JS 事件流机制详解:冒泡、捕获与完整事件流

文章目录

  • JS 事件流机制详解:冒泡、捕获与完整事件流
    • 一、DOM 事件流基本概念
    • 二、事件捕获 (Event Capturing)
      • 特点
      • 代码示例
    • 三、事件冒泡 (Event Bubbling)
      • 特点
      • 代码示例
    • 四、完整事件流示例
      • HTML 结构
      • JavaScript 代码
    • 五、事件流控制方法
      • 1. 阻止事件传播
      • 2. 阻止默认行为
      • 3. 事件委托 (利用冒泡)
    • 六、实际应用场景
    • 七、注意事项

一、DOM 事件流基本概念

DOM 事件流描述了事件在网页中传播的完整过程,包含三个阶段:

  1. 捕获阶段 (Capture Phase):从 window 对象向下传播到目标元素

  2. 目标阶段 (Target Phase):事件到达目标元素

  3. 冒泡阶段 (Bubble Phase):从目标元素向上传播回 window 对象
    在这里插入图片描述

二、事件捕获 (Event Capturing)

特点

  • 传播方向:从最外层祖先元素向目标元素传播
  • 触发顺序:window → document → html → body → … → 目标元素
  • 应用场景:需要在事件到达目标前拦截处理

代码示例

// 添加捕获阶段事件监听 (第三个参数为true)
document.querySelector('#outer').addEventListener('click', function() {
    console.log('捕获阶段 - Outer');
}, true);

document.querySelector('#inner').addEventListener('click', function() {
    console.log('捕获阶段 - Inner');
}, true);

三、事件冒泡 (Event Bubbling)

特点

  • 传播方向:从目标元素向最外层祖先元素传播
  • 触发顺序:目标元素 → … → body → html → document → window
  • 默认行为:大多数原生事件都会冒泡(focus/blur等除外)

代码示例

// 添加冒泡阶段事件监听 (第三个参数为false或省略)
document.querySelector('#outer').addEventListener('click', function() {
    console.log('冒泡阶段 - Outer');
});

document.querySelector('#inner').addEventListener('click', function() {
    console.log('冒泡阶段 - Inner');
});

四、完整事件流示例

HTML 结构

<div id="grandparent">
    Grandparent
    <div id="parent">
        Parent
        <div id="child">Child</div>
    </div>
</div>

JavaScript 代码

const elements = ['grandparent', 'parent', 'child'];

// 为所有元素添加捕获和冒泡监听器
elements.forEach(id => {
    const el = document.getElementById(id);
    
    // 捕获阶段
    el.addEventListener('click', () => {
        console.log(`捕获阶段: ${id}`);
    }, true);
    
    // 冒泡阶段
    el.addEventListener('click', () => {
        console.log(`冒泡阶段: ${id}`);
    }, false);
});

// 点击child元素后的输出顺序:
// 捕获阶段: grandparent
// 捕获阶段: parent
// 捕获阶段: child (实际是目标阶段)
// 冒泡阶段: child  (实际是目标阶段)
// 冒泡阶段: parent
// 冒泡阶段: grandparent

五、事件流控制方法

1. 阻止事件传播

element.addEventListener('click', function(event) {
    event.stopPropagation(); // 阻止继续传播
    // event.stopImmediatePropagation(); // 阻止所有后续监听器执行
});

2. 阻止默认行为

element.addEventListener('click', function(event) {
    event.preventDefault(); // 阻止默认行为(如链接跳转)
});

3. 事件委托 (利用冒泡)

// 只在父元素上设置监听器,处理子元素事件
document.querySelector('#parent').addEventListener('click', function(event) {
    if(event.target.id === 'child') {
        console.log('通过冒泡处理child点击');
    }
});

六、实际应用场景

  1. 事件委托:利用冒泡减少事件监听器数量

    // 动态列表项处理
    document.querySelector('#list').addEventListener('click', function(event) {
        if(event.target.classList.contains('item')) {
            console.log('点击了项目:', event.target.textContent);
        }
    });
    
  2. 拦截处理:使用捕获阶段提前处理事件

    // 全局点击拦截
    document.addEventListener('click', function(event) {
        if(event.target.tagName === 'A') {
            console.log('即将跳转到:', event.target.href);
        }
    }, true);
    
  3. 自定义组件:控制组件内外事件传播

    // 阻止组件内部事件冒泡到外部
    customComponent.addEventListener('click', function(event) {
        event.stopPropagation();
        // 组件内部处理逻辑
    });
    

七、注意事项

  1. 目标阶段特殊性
    • 在目标元素上,捕获和冒泡监听器按注册顺序执行
    • 不属于真正的捕获或冒泡阶段
  2. 不冒泡的事件
    • focus/blur
    • load/unload
    • mouseenter/mouseleave
  3. 性能考虑
    • 过多的事件监听器会影响性能
      目标阶段特殊性
    • 在目标元素上,捕获和冒泡监听器按注册顺序执行
    • 不属于真正的捕获或冒泡阶段
  4. 不冒泡的事件
    • focus/blur
    • load/unload
    • mouseenter/mouseleave
  5. 性能考虑
    • 过多的事件监听器会影响性能
    • 合理使用事件委托优化

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

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

相关文章

算法专题七:分治

快排 1.颜色分类 题目链接:75. 颜色分类 - 力扣(LeetCode) class Solution {public void swap(int[] nums, int i, int j){int t = nums[i];nums[i] = nums[j];nums[j] = t;}public void sortColors(int[] nums) {int left=-1 ,i=0 ,right=nums.length;while(i<right){i…

Vue中虚拟DOM的原理与作用

绪论 首先我们先了解&#xff0c;DOM&#xff08;Document Object Model&#xff0c;文档对象模型&#xff09; 是浏览器对 HTML/XML 文档的结构化表示&#xff0c;它将文档解析为一个由节点&#xff08;Node&#xff09;和对象组成的树形结构&#xff08;称为 DOM 树&#xf…

使用vue3+ts+input封装上传组件,上传文件显示文件图标

效果图&#xff1a; 代码 <template><div class"custom-file-upload"><div class"upload"><!-- 显示已选择的文件 --><div class"file-list"><div v-for"(item, index) in state.filsList" :key&q…

【Linux】Ubuntu 创建应用图标的方式汇总,deb/appimage/通用方法

Ubuntu 创建应用图标的方式汇总&#xff0c;deb/appimage/通用方法 对于标准的 Ubuntu&#xff08;使用 GNOME 桌面&#xff09;&#xff0c;desktop 后缀的桌面图标文件主要保存在以下三个路径&#xff1a; 当前用户的桌面目录&#xff08;这是最常见的位置&#xff09;。所…

LangGraph--Agent工作流

Agent的工作流 下面展示了如何创建一个“计划并执行”风格的代理。 这在很大程度上借鉴了 计划和解决 论文以及Baby-AGI项目。 核心思想是先制定一个多步骤计划&#xff0c;然后逐项执行。完成一项特定任务后&#xff0c;您可以重新审视计划并根据需要进行修改。 般的计算图如…

Spring Boot 常用注解面试题深度解析

&#x1f91f;致敬读者 &#x1f7e9;感谢阅读&#x1f7e6;笑口常开&#x1f7ea;生日快乐⬛早点睡觉 &#x1f4d8;博主相关 &#x1f7e7;博主信息&#x1f7e8;博客首页&#x1f7eb;专栏推荐&#x1f7e5;活动信息 文章目录 Spring Boot 常用注解面试题深度解析一、核心…

Linux系统的CentOS7发行版安装MySQL80

文章目录 前言Linux命令行内的”应用商店”安装CentOS的安装软件的yum命令安装MySQL1. 配置yum仓库2. 使用yum安装MySQL3. 安装完成后&#xff0c;启动MySQL并配置开机自启动4. 检查MySQL的运行状态 MySQL的配置1. 获取MySQL的初始密码2. 登录MySQL数据库系统3. 修改root密码4.…

408第一季 - 数据结构 - 栈与队列

栈 闲聊 栈是一个线性表 栈的特点是后进先出 然后是一个公式 比如123要入栈&#xff0c;一共有5种排列组合的出栈 栈的数组实现 这里有两种情况&#xff0c;&#xff0c;一个是有下标为-1的&#xff0c;一个没有 代码不用看&#xff0c;真题不会考 栈的链式存储结构 L ->…

【RTP】Intra-Refresh模式下的 H.264 输出,RTP打包的方式和普通 H.264 流并没有本质区别

对于 Intra-Refresh 模式下的 H.264 输出,RTP 打包 的方式和普通 H.264 流并没有本质区别:你依然是在对一帧一帧的 NAL 单元进行 RTP 分包,只不过这些 NAL 单元内部有部分宏块是 “帧内编码” 而已。下面分步骤说明: 1. 原理回顾:RFC 6184 H.264 over RTP 按照 RFC 6184 …

Redis实战-消息队列篇

前言&#xff1a; 讲讲做消息队列遇到的问题。 今日所学&#xff1a; 异步优化消息队列基于stream实现异步下单 1. 异步优化 1.1 需求分析 1.1.1 现有下单流程&#xff1a; 1.查询优惠劵 2.判断是否是秒杀时间&#xff0c;库存是否充足 3.实现一人一单 在这个功能中&…

(三)Linux性能优化-CPU-CPU 使用率

CPU使用率 user&#xff08;通常缩写为 us&#xff09;&#xff0c;代表用户态 CPU 时间。注意&#xff0c;它不包括下面的 nice 时间&#xff0c;但包括了 guest 时间。nice&#xff08;通常缩写为 ni&#xff09;&#xff0c;代表低优先级用户态 CPU 时间&#xff0c;也就是进…

佰力博科技与您探讨材料介电性能测试的影响因素

1、频率依赖性 材料的介电性能通常具有显著的频率依赖性。在低频下&#xff0c;偶极子的取向极化占主导&#xff0c;介电常数较高&#xff1b;而在高频下&#xff0c;偶极子的取向极化滞后&#xff0c;导致介电常数下降&#xff0c;同时介电损耗增加。例如&#xff0c;VHB4910…

K8S认证|CKS题库+答案| 4. RBAC - RoleBinding

目录 4. RBAC - RoleBinding 免费获取并激活 CKA_v1.31_模拟系统 题目 开始操作&#xff1a; 1&#xff09;、切换集群 2&#xff09;、查看SA和role 3&#xff09;、编辑 role-1 权限 4&#xff09;、检查role 5&#xff09;、创建 role和 rolebinding 6&#xff0…

React 新项目

使用git bash 创建一个新项目 建议一开始就创建TS项目 原因在Webpack中改配置麻烦 编译方法:ts compiler 另一种 bable 最好都配置 $ create-react-app cloundmusic --template typescript 早期react项目 yarn 居多 目前npm包管理居多 目前pnpm不通用 icon 在public文件夹中…

AI编程在BOSS项目的实践经验分享

前言 在人工智能技术革新浪潮的推动下&#xff0c;智能编程助手正以前所未有的速度重塑开发领域。这些基于AI的代码辅助工具通过智能提示生成、实时错误检测和自动化重构等功能&#xff0c;显著提升了软件工程的全流程效率。无论是初入行业的开发者还是资深程序员&#xff0c;…

数学:”度量空间”了解一下?

度量空间是现代数学中一种基本且重要的抽象空间。以下是对它的详细介绍&#xff1a; 定义 相关概念 常见的度量空间举例 度量空间的类型 度量空间的作用 度量空间是拓扑空间的一种特殊情况&#xff0c;它为拓扑空间的研究提供了具体的模型和实例。同时&#xff0c;度量空间在…

jenkins脚本查看及备份

位置与备份 要完整备份 Jenkins 的所有脚本和相关配置&#xff0c;包括 Jenkinsfile、构建脚本&#xff08;如 .sh / .bat&#xff09;、Job 配置、插件、凭据等&#xff0c;你可以从两个层面入手&#xff1a; ✅ 一、完整备份 Jenkins 主目录&#xff08;最全面&#xff09; …

用电脑通过网口控制keysight示波器

KEYSIGHT示波器HD304MSO性能 亮点: 体验 200 MHz 至 1 GHz 的带宽和 4 个模拟通道。与 12 位 ADC 相比,使用 14 位模数转换器 (ADC) 将垂直分辨率提高四倍。使用 10.1 英寸电容式触摸屏轻松查看和分析您的信号。捕获 50 μVRMS 本底噪声的较小信号。使用独有区域触摸在几秒…

《如何使用MinGW-w64编译OpenCV和opencv_contrib》

《如何使用MinGW-w64编译OpenCV和opencv_contrib》 在Windows环境下使用MinGW编译OpenCV和opencv_contrib是一个常见需求,尤其是对于那些希望使用GCC工具链而非Visual Studio的开发者。下面我将详细介绍这个过程。 准备工作 首先需要安装和准备以下工具和库: MinGW(建议使…

模拟搭建私网访问外网、外网访问服务器服务的实践操作

目录 实验环境 实践要求 一、准备工作 1、准备四台虚拟机&#xff0c;分别标号 2、 防火墙额外添加两块网卡&#xff0c;自定义网络连接模式 3、 关闭虚拟机的图形管理工具 4、关闭防火墙 5、分别配置四台虚拟机的IP地址&#xff0c;此处举一个例子&#xff08;使用的临…