JavaScript事件触发过程底层原理

在Web开发中,JavaScript的事件处理是一个至关重要的主题。理解事件触发的过程以及掌握事件代理是成为优秀前端工程师的关键之一。本文将深入探讨JavaScript中事件的触发过程以及事件代理的概念、优势和使用场景。


首先我们有着这样三个互相嵌套的div,a,b,c,a为红色,b为蓝色,c为黑色。 接着给a,b,c都添加上点击事件

事件触发过程

js 复制代码
 a.addEventListener("click",()=>{
        console.log('a被点击');
    },false)

    b.addEventListener("click",(event)=>{
        console.log('b被点击');
    },false)

    c.addEventListener("click",(event)=>{
        console.log('c被点击');
    },false)

显然此时点击最内层的c三个点击事件都会被触发。但是!!!

这三个事件触发的顺序是怎么样的以及为什么是这样

首先告诉大家

事件触发的过程是c,b,a,那么为什么是这样一个过程呢?

背后的底层原理

在JavaScript中的事件触发过程分为捕获过程和冒泡过程,这两个过程构成了事件传播的整体流程。

  1. 捕获过程:window往事件触发处传播。
  2. 目标触发: 传播到事件目标处,触发注册的事件处理函数。
  3. 冒泡过程: 从事件触发处往window上传播。当遇到注册的冒泡事件时,触发相应的处理函数。这个过程是默认的事件触发方式。 点击c整个事件流的传播过程是这样的:
  4. window->c->b->a捕获过程
  5. a->b->c->window冒泡过程 a,b,c绑定的事件就在冒泡过程依次被触发 需要注意的是,捕获事件和冒泡事件是互斥的,在捕获事件内触发了就不会在冒泡事件内触发。而且在JavaScript中,默认所有事件都只在冒泡过程中触发。如果想要在捕获过程中触发,可以通过修改addEventListener的第三个参数来实现

这个值默认是false,当为false时就会在冒泡过程时触发,如果设置为true就会在捕获过程时触发。

阻止默认行为

在事件处理中,有时候需要阻止事件的默认行为,比如阻止链接的跳转或表单的提交。为此,可以使用以下方法:

  1. event.stopPropagation() 该方法可以阻止事件流的传播,即停止事件在捕获或冒泡阶段的传递。
  2. event.stopImmediatePropagation() 除了阻止事件流传播外,该方法还能阻止同一容器上绑定的其他相同事件的执行。
js 复制代码
   b.addEventListener("click",(event)=>{
        console.log('b被点击');
        event.stopPropagation()
    },true)

如果设置b身上绑定的事件在捕获过程时触发并在触发的事件内部添加event.stopPropagation(),此时点击c就会输出如下

此时在捕获过程由window向c传播,c------>b然后此时b事件触发并且阻断事件流的传播。至此事件流的传播就结束了。

这两个方法之间的区别在于,event.stopImmediatePropagation()还能够阻断自身的事件传播。当我们在c身上添加两个点击事件并给第一个点击事件加入 event.stopPropagation();时,此时我们点击c就会发现

js 复制代码
    c.addEventListener("click",(event)=>{
        console.log('c被点击');
        event.stopPropagation();
    },false)

    c.addEventListener("click",(event)=>{
        console.log('c被点击2');
    },false)

会有着两次输出。如果此时把第一个点击事件中的event.stopPropagation();换成event.stopImmediatePropagation()我们就会发现 只有着一次输出了,这就是因为event.stopImmediatePropagation()还会阻断自身事件流的传播,不过需要注意的是,写有event.stopImmediatePropagation()的点击事件需要在另外一个点击事件的上面才能够被阻断。

这两个方法的灵活运用可以有效地控制事件的传播行为。

相关推荐
aa小小34 分钟前
localhost 访问异常排查笔记
前端
格子软件35 分钟前
2026年GEO优化系统源码的分布式状态机深度拆解
java·前端·vue.js·vue·geo
陈随易1 小时前
Rust、Golang、MoonBit 编译成 WASM,体积和速度差距有多大?
前端·后端·程序员
IT_陈寒1 小时前
Python多线程的坑,我居然现在才踩到
前端·人工智能·后端
摇滚侠1 小时前
方法 A 等方法 B 执行完再执行 叫同步调用还是异步调用 JS 默认是同步调用还是异步调用
开发语言·javascript·ecmascript
触底反弹2 小时前
🔥 字符串算法面试三连击:反转、回文、回文变种,搞懂这三题稳了!
前端·javascript·算法
触底反弹2 小时前
AI Tool Use 深度解析:大模型是如何"突破物理限制"调用外部工具的?
javascript·人工智能·后端
竹林8182 小时前
从 RPC 超时到批量签名:我用 @solana/web3.js 重构了一个 NFT 铸造页面,踩了这些坑
前端·javascript
工业HMI实战笔记2 小时前
工业HMI界面布局“1核2辅”黄金结构,适配90%场景
前端·ui·性能优化·自动化·交互
优雅格子衫3 小时前
TypeScript 类的基本使用小结
javascript·ubuntu·typescript