JS中给元素添加事件监听器的各种方法详解(包含比较和应用场景)

news2024/9/16 9:08:58

JavaScript 中给元素添加事件监听器的各种方法详解

在 JavaScript 中,事件处理是前端开发的一个重要部分。无论是点击按钮、提交表单,还是鼠标悬停,都涉及到事件监听。本文中,我将详细讲解各种给元素添加事件监听器的方法,包括每种方法的优缺点、实际示例、使用场景及彼此的对比。

1. 使用 HTML 中的事件属性(Inline Event Handlers)

这是早期最直接的事件绑定方式,直接在 HTML 元素的标签中指定事件处理程序。

示例:
<button onclick="handleClick()">Click me</button>

<script>
  function handleClick() {
    alert('Button clicked!');
  }
</script>
优点:
  • 简单直接:事件逻辑直接嵌入到 HTML 标签中,易于理解,特别是初学者快速上手。
  • 快速实现交互:对于简单的交互,可以快速实现。
缺点:
  • 污染 HTML:JavaScript 代码直接嵌入到 HTML 中,破坏了代码的分离性,不利于维护和复用。
  • 只能绑定一个事件处理程序:因为事件处理程序直接定义在标签属性中,只能绑定一个处理程序,无法绑定多个。
  • 难以维护:当项目变得复杂时,内联事件会让 HTML 和逻辑混合在一起,维护困难。
使用场景:

适用于非常简单或一次性的功能,快速展示页面交互功能时可用。


2. 使用 JavaScript 中的 DOM 元素的事件属性

这种方式是在 JavaScript 中直接通过 DOM 元素的事件属性(如 onclick)来绑定事件处理程序。事件处理函数是在 JavaScript 逻辑中,而不是嵌入到 HTML 标签中。

示例:
<button id="myButton">Click me</button>

<script>
  const btn = document.getElementById('myButton');
  btn.onclick = function() {
    alert('Button clicked!');
  };
</script>
优点:
  • 与 HTML 代码分离:JavaScript 逻辑与 HTML 结构分离,代码稍微更干净,便于理解。
  • 简单易懂:直接通过元素属性来添加事件,与内联事件处理类似,但更具灵活性。
缺点:
  • 覆盖问题:对于同一个事件类型,只能绑定一个处理程序。后绑定的处理程序会覆盖之前绑定的。例如,绑定多个 onclick 处理程序时,只会保留最后一个。

    覆盖示例:

    const btn = document.getElementById('myButton');
    btn.onclick = function() {
      alert('First handler');
    };
    
    // 覆盖了上面的事件处理程序
    btn.onclick = function() {
      alert('Second handler');
    };
    
  • 不支持事件委托:无法通过事件冒泡来处理动态生成的子元素的事件。

使用场景:

适用于简单的项目,且每个元素只需绑定一个事件处理程序的场景。


3. 使用 addEventListener() 方法

这是现代浏览器推荐的事件绑定方式。addEventListener() 方法不仅可以为同一元素添加多个事件监听器,还可以控制事件的捕获和冒泡行为。相比 onclick 等直接绑定事件的方法,它提供了更强的灵活性。

基本使用示例:
<button id="myButton">Click me</button>

<script>
  const btn = document.getElementById('myButton');
  
  btn.addEventListener('click', function() {
    alert('Button clicked!');
  });
</script>
绑定多个事件处理程序:

addEventListener() 允许为同一元素的同一事件类型添加多个事件处理程序。

const btn = document.getElementById('myButton');

// 第一个事件处理程序
btn.addEventListener('click', function() {
  alert('First handler');
});

// 第二个事件处理程序
btn.addEventListener('click', function() {
  alert('Second handler');
});

// 点击按钮时,依次触发两个事件处理程序
解除事件监听器:

可以使用 removeEventListener() 来移除绑定的事件处理程序。注意,移除事件处理程序时,必须传递与绑定时完全相同的函数引用,匿名函数无法移除。

const btn = document.getElementById('myButton');

function handleClick() {
  alert('Button clicked!');
}

// 绑定事件
btn.addEventListener('click', handleClick);

// 解除事件绑定
btn.removeEventListener('click', handleClick);
第三个参数(控制事件捕获与冒泡):

addEventListener() 的第三个参数可以是一个布尔值或对象,用来控制事件是在捕获阶段还是冒泡阶段触发。

  • 事件捕获:从根元素开始向目标元素传播。
  • 事件冒泡:从目标元素开始向根元素传播(这是默认行为)。

捕获阶段示例:

const btn = document.getElementById('myButton');

// 捕获阶段处理事件
btn.addEventListener('click', function() {
  alert('Captured click event');
}, true); // 第三个参数为 true 表示捕获阶段

对象作为第三个参数:

btn.addEventListener('click', function() {
  alert('Button clicked with options');
}, { capture: true, once: true });
  • capture: true:表示在捕获阶段触发事件。
  • once: true:表示事件处理程序仅执行一次后自动移除。
优点:
  • 支持多个事件处理程序:同一事件类型可以绑定多个事件处理程序,按顺序依次执行。
  • 细粒度控制:可以通过第三个参数控制事件是在捕获还是冒泡阶段触发。
  • 可移除监听器:支持使用 removeEventListener() 解除事件监听。
  • 支持事件委托:可通过事件冒泡机制来进行事件委托,处理动态生成的子元素事件。
缺点:
  • 老旧浏览器兼容性:在非常老的浏览器(如 IE8 及以下)中不支持,需使用 attachEvent() 替代。
使用场景:

适用于现代 Web 开发的所有场景,尤其适合需要为同一元素绑定多个事件处理程序、或者需要精细控制事件阶段的情况。


4. 事件委托(Event Delegation)

事件委托是一种优化事件处理的技术,它利用事件冒泡机制,通过在父级元素上绑定事件监听器,来处理多个子元素的事件。对于动态生成的子元素或大量子元素来说,事件委托能显著提升性能和代码简洁度。

示例:
<ul id="parent">
  <li>Item 1</li>
  <li>Item 2</li>
  <li>Item 3</li>
</ul>

<script>
  const parent = document.getElementById('parent');
  
  parent.addEventListener('click', function(event) {
    if (event.target.tagName === 'LI') {
      alert('You clicked: ' + event.target.textContent);
    }
  });
</script>

在这个例子中,只需在父级 ul 元素上绑定一个点击事件监听器,就可以处理所有 li 元素的点击事件。这对于动态添加的 li 元素同样有效。

优点:
  • 性能优化:通过减少绑定的事件处理程序数量,降低内存消耗,提升性能,特别是对于动态生成的大量子元素。
  • 动态元素支持:即使是通过 JavaScript 动态添加的元素,也能通过事件委托进行事件处理。
  • 代码简洁:避免为每个子元素分别绑定事件处理程序。
缺点:
  • 不适用于不冒泡的事件:例如 focusblur 事件,这些事件不会冒泡,因此不能通过事件委托来处理。
  • 需要手动筛选目标:事件冒泡时,父级元素会捕捉到所有子元素的事件,开发者需要在事件处理程序中筛选 event.target 来确定触发的具体元素。
使用场景:

适用于大量子元素事件处理,或者需要动态生成子元素的场景。例如表格、列表等大量 DOM 结构的事件处理。


5. attachEvent() 方法(仅用于 IE8 及以下)

这是旧版 IE 浏览器的事件绑定方法,与标准的 addEventListener() 类似,但有一些限制。

示例:
const btn = document.getElementById('myButton');
btn.attachEvent('onclick', function() {
  alert('Button clicked!');
});
优点:
  • 兼容老旧浏览器:这是为了支持 IE8 及以下版本的旧式浏览器。
缺点:
  • 非标准attachEvent() 是 IE 专有的事件处理方式,不符合标准,现代浏览器不支持。
  • **无法控制事件的捕获

和冒泡**:attachEvent() 无法像 addEventListener() 一样灵活控制事件的捕获和冒泡。

  • 上下文问题:在 attachEvent 中,this 关键字总是指向 window 对象,而不是事件的目标元素,导致代码不一致。
使用场景:

仅用于需要兼容 IE8 及以下浏览器的项目中。但在现代开发中,通常不推荐使用。


总结

方法优点缺点使用场景
HTML 内联事件简单、直观代码难维护,无法绑定多个处理程序非常简单的页面或一次性项目
DOM 事件属性简单、与 HTML 分离覆盖问题,无法绑定多个处理程序简单页面交互
addEventListener()支持多个处理程序,解除事件监听,事件委托,控制捕获和冒泡需要考虑老旧浏览器兼容性现代 Web 开发,复杂交互场景
事件委托性能好,支持动态元素不适用于不冒泡的事件,手动筛选目标动态子元素或大量子元素的事件处理
attachEvent()兼容旧版 IE非标准,事件处理不一致仅用于兼容 IE8 及以下

结论

在现代 Web 开发中,推荐使用 addEventListener(),它功能强大且灵活,适用于几乎所有的事件处理场景。而事件委托则是处理大量动态生成元素的最佳实践。对于非常简单的项目,可以考虑使用 DOM 事件属性。而内联事件处理和 attachEvent() 则不建议在现代开发中使用。

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

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

相关文章

C语言 | Leetcode C语言题解之第390题消除游戏

题目&#xff1a; 题解&#xff1a; int lastRemaining(int n) {int a1 1;int k 0, cnt n, step 1;while (cnt > 1) {if (k % 2 0) { // 正向a1 a1 step;} else { // 反向a1 (cnt % 2 0) ? a1 : a1 step;}k;cnt cnt >> 1;step step << 1;}return …

【机器学习】和【人工智能】在量子力学的应用及代码案例分析

知孤云出岫 这里写目录标题 一、机器学习和人工智能在量子力学中的应用概述二、量子态的表示与模拟2.1 变分自编码器&#xff08;VAE&#xff09;用于量子态模拟 三、量子系统的哈密顿量学习3.1 使用机器学习推断哈密顿量 四、量子计算中的算法优化4.1 变分量子算法&#xff08…

AI大模型日报#0908:OpenAI计划年底推出GPT Next、Roblox官宣AI秒生3D物体模型

导读&#xff1a;AI大模型日报&#xff0c;爬虫LLM自动生成&#xff0c;一文览尽每日AI大模型要点资讯&#xff01;目前采用“文心一言”&#xff08;ERNIE-4.0-8K-latest&#xff09;、“智谱AI”&#xff08;glm-4-0520&#xff09;生成了今日要点以及每条资讯的摘要。欢迎阅…

如果文件从存储卡中被误删除,存储卡数据恢复如何恢复?

如果文件被误从存储卡中删除&#xff0c;如何从Android 手机、相机或其他数码设备&#xff08; SD/CF/MicroSD/xD ...&#xff09;恢复照片视频&#xff1b;存储卡已格式化&#xff0c;原始 0 字节&#xff0c;空白&#xff1b;存储卡要求格式化&#xff1b;存储卡未显示、无法…

基于ONNX-YOLOv10-Object-Detection项目实现yolov10模型onnx-python推理

项目地址&#xff1a;https://github.com/ibaiGorordo/ONNX-YOLOv10-Object-Detection 项目依赖&#xff1a;onnxruntime-gpu、opencv-python、imread-from-url、cap-from-youtube、ultralytics 1、代码修改 代码改动说明&#xff1a;yolov10/yolov10.py中的第18行修改为以下…

Docker部署tenine实现后端应用的高可用与负载均衡

采用Docker方式的Tengine 和 keepalived 组合模式可以实现小应用场景的高可用负载均衡需求 目录 网络架构一、环境准备二、软件安装1. 下载Tenine镜像2. 下载Keepalived镜像3. 制作SpringBoot镜像 三、软件配置1. 创建应用容器2. 代理访问应用3. 创建Keepalived4. 测试高可用 网…

基于YOLOv5的积水检测模型训练:从数据到模型的全面解析

之前给大家带来了Yololov5Pyqt5Opencv 实时城市积水报警系统&#xff0c; 详见&#xff1a; Yololov5Pyqt5Opencv 实时城市积水报警系统_yolo opencv pyqt5-CSDN博客 今天详细解析一下积水检测模型训练部分的内容 在积水检测项目中&#xff0c;实时性和准确性是至关重要的。…

STM32内部闪存FLASH(内部ROM)、IAP

1 FLASH简介 1 利用程序存储器的剩余空间来保存掉电不丢失的用户数据 2 通过在程序中编程(IAP)实现程序的自我更新 &#xff08;OTA&#xff09; 3在线编程&#xff08;ICP把整个程序都更新掉&#xff09; 1 系统的Bootloader写死了&#xff0c;只能用串口下载到指定的位置&a…

Spring Boot事务管理

事务管理 事务进阶 如果在删除了部门之后&#xff0c;出现了异常。那么就会出现部门被删除之后其中的员工并未被删除。 Transactional注解&#xff0c;在事务执行完成之后自动提交或者回滚。只需要在执行多次数据修改的事务上加上该注解即可。&#xff08;比如两次Update或者…

嵌入式Openharmony系统构建与启动详解

大家好,今天主要给大家分享一下,如何构建Openharmony子系统以及系统的启动过程分解。 第一:OpenHarmony系统构建 首先熟悉一下,构建系统是一种自动化处理工具的集合,通过将源代码文件进行一系列处理,最终生成和用户可以使用的目标文件。这里的目标文件包括静态链接库文件…

基于51单片机的自动转向修复系统的设计与实现

文章目录 前言资料获取设计介绍功能介绍设计清单具体实现截图参考文献设计获取 前言 &#x1f497;博主介绍&#xff1a;✌全网粉丝10W,CSDN特邀作者、博客专家、CSDN新星计划导师&#xff0c;一名热衷于单片机技术探索与分享的博主、专注于 精通51/STM32/MSP430/AVR等单片机设…

11.5.软件系统分析与设计-面向对象的程序设计与实现

面向对象的程序设计与实现 设计模式 Java代码 C代码

Android13_SystemUI下拉框新增音量控制条

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 Android13_SystemUI下拉框新增音量控制条 一、必备知识二、源码分析对比1.brightness模块分析对比2.statusbar/phone 对应模块对比对比初始化类声明对比构造方法 三、源码修改…

操作系统week1

操作系统学习 一.操作系统概述 1.概念、功能 操作系统是指控制和管理整个计算机系统的硬件和软件资源&#xff0c;并合理地组织调度计算机的工作和资源的分配&#xff0c;以提供给用户和其他软件方便的接口和环境 #mermaid-svg-SpFSwhrPg2GwVnYt {font-family:"trebuch…

【Python 千题 —— 算法篇】数字反转

Python 千题持续更新中 …… 脑图地址 &#x1f449;&#xff1a;⭐https://twilight-fanyi.gitee.io/mind-map/Python千题.html⭐ 题目背景 整数反转是一个经典的算法问题&#xff0c;常见于各种编程竞赛和技术面试中。它要求我们将给定的整数按位进行翻转&#xff0c;并返…

c++ string类的模拟实现的注意事项

一.构造函数 第一种形式&#xff0c;使用字符指针赋值 为了防止修改&#xff0c;我们传入了常量字符串。但是这里的初始化列表出错了&#xff0c;因为_str是一个变量&#xff0c;将常量给到一个变量涉及到权限的放大&#xff0c;是错误的。那该怎么写呢&#xff1f;对_str的赋…

证书学习(四)X.509数字证书整理

目录 一、X.509证书 介绍1.1 什么是 X.509证书?1.2 什么是 X.509标准?1.3 什么是 PKI?二、X.509证书 工作原理2.1 PKI 的基础——加密算法2.2 PKI 证书编码三、X.509证书 结构3.1 证书字段3.2 证书扩展背景: 我们在日常的开发过程中,经常会遇到各种各样的电子证书文件,其…

【每日一题】LeetCode 2181.合并零之间的节点(链表、模拟)

【每日一题】LeetCode 2181.合并零之间的节点&#xff08;链表、模拟&#xff09; 题目描述 给定一个链表&#xff0c;链表中的每个节点代表一个整数。链表中的整数由 0 分隔开&#xff0c;表示不同的区间。链表的开始和结束节点的值都为 0。任务是将每两个相邻的 0 之间的所…

虚拟机VMware桥接网络命令来重置 /etc/sysconfig/network-scripts/ifcfg-ens33 文件

ifcfg-ens33 文件专门用于配置这个特定的网络接口。 有时候把ifcfg-ens33文件配置弄乱了&#xff0c;可以使用命令重置。 最常用的方式是通过 nmcli 或者 nmtui 来重置网络接口的配置。 第一种方法. 使用 nmcli 命令重置网络配置 nmcli 是一个强大的网络管理命令行工具&…

闯关leetcode——3.Longest Substring Without Repeating Characters

大纲 题目地址内容 解题代码地址 题目 地址 https://leetcode.com/problems/longest-substring-without-repeating-characters/description/ 内容 Given a string s, find the length of the longest substring without repeating characters. Example 1: Input: s “abc…