JS DOM操作与事件处理从入门到实践

news2025/5/11 10:30:54

对于前端开发者来说,让静态的 HTML 页面变得生动、可交互是核心技能之一。实现这一切的关键在于理解和运用文档对象模型 (DOM) 以及 JavaScript 的事件处理机制。本文将带你深入浅出地探索 DOM 操作的奥秘,并掌握JavaScript 事件处理的方方面面。

目录

Part 1: 文档对象模型 (DOM) 揭秘

1. 什么是 DOM?

2. DOM 树结构

Part 2: 核心 DOM 操作:查询、创建、修改、删除

1. 查询 DOM 元素 ("查")

2. 创建新 DOM 元素 ("增")

3. 添加/插入 DOM 元素 ("增")

4. 修改 DOM 元素 ("改")

5. 删除 DOM 元素 ("删")

6. 替换 DOM 元素 ("改")

Part 3: JavaScript 事件处理机制

1. 什么是事件?

2. 事件监听器 

3. 事件对象 

4. 事件流:捕获与冒泡的奇妙旅程

5. 事件委托/代理:聪明的事件管理员

6. 常见事件类型

Part 4: 实践与最佳做法:写出高效、优雅的 DOM 与事件代码

1. 提升性能的考量 

2. 提升代码组织与可读性 

3. 关注可访问性 (Accessibility - a11y)


Part 1: 文档对象模型 (DOM) 揭秘

1. 什么是 DOM?

DOM (Document Object Model) 即文档对象模型,是浏览器为 HTML 或 XML 文档创建的一个编程接口。简单来说,当浏览器加载一个 HTML 页面时,它会解析 HTML 代码并创建一个树状的数据结构来表示这个页面——这个树就是 DOM。

  • 桥梁作用: DOM 是 HTML 文档与 JavaScript 之间沟通的桥梁。JavaScript 不能直接理解原始的 HTML 文本,但它可以理解和操作 DOM 这个对象集合。
  • 树形结构: DOM 将文档中的每一个部分(如元素、文本、注释等)都视为一个节点 (Node)。这些节点按照它们在 HTML 中的层级关系组织起来,形成一个树形结构。
  • 节点类型: 主要有元素节点 (Element Node)、文本节点 (Text Node)、属性节点 (Attribute Node) 和注释节点 (Comment Node)。我们最常打交道的是元素节点和文本节点。
2. DOM 树结构

想象一下你的 HTML 文档是一棵家谱树:

  • <html> 标签是树根。
  • <head><body><html> 的直接子节点。
  • <body> 内部的 <div>, <p>, <ul> 等元素是其子节点,它们之间也可能存在父子或兄弟关系。

document 对象是整个 DOM 树的入口点,代表了整个文档。我们可以通过 document 对象来访问和操作页面上的任何元素。

HTML 示例:
<html>
  <head>
    <title>我的页面</title>
  </head>
  <body>
    <header>
      <h1>欢迎!</h1>
    </header>
    <main>
      <p>这是一个段落。</p>
      <button id="myButton">点我</button>
    </main>
  </body>
</html>

DOM 树 (简化表示):
document
  └── html
      ├── head
      │   └── title
      │       └── "我的页面" (文本节点)
      └── body
          ├── header
          │   └── h1
          │       └── "欢迎!" (文本节点)
          └── main
              ├── p
              │   └── "这是一个段落。" (文本节点)
              └── button (id="myButton")
                  └── "点我" (文本节点)

Part 2: 核心 DOM 操作:查询、创建、修改、删除

JavaScript 提供了丰富的 API 来对 DOM 进行增、删、改、查操作。

1. 查询 DOM 元素 ("查")

获取页面上已存在的元素是进行后续操作的第一步。

  • document.getElementById('id'): 通过元素的 id 属性获取单个元素。ID 在文档中应该是唯一的。
    const myButton = document.getElementById('myButton');
    console.log(myButton); // 输出 <button id="myButton">...</button>
    
  • document.getElementsByClassName('className'): 通过元素的 class 名称获取一个HTMLCollection (元素的集合)。
    const items = document.getElementsByClassName('list-item'); // 假设页面有多个 class="list-item" 的元素
    for (let i = 0; i < items.length; i++) {
      console.log(items[i]);
    }
    
  • document.getElementsByTagName('tagName'): 通过元素的标签名 (如 'p', 'div') 获取一个HTMLCollection。 
    const paragraphs = document.getElementsByTagName('p');
    console.log(paragraphs.length); // 输出页面中 <p> 元素的数量
    
  • document.querySelector('cssSelector'): 使用 CSS 选择器语法获取匹配到的第一个元素。非常强大和灵活。 
    const firstListItem = document.querySelector('ul li.active'); // 获取 ul 下第一个 class 为 active 的 li
    const header = document.querySelector('#main-header h1');
    console.log(header);
    
  • document.querySelectorAll('cssSelector'): 使用 CSS 选择器语法获取匹配到的所有元素,返回一个NodeList (静态的节点集合)。 
    const allLinks = document.querySelectorAll('a[target="_blank"]'); // 获取所有 target="_blank" 的链接
    allLinks.forEach(link => {
      console.log(link.href);
    });
    
    注意:
    • HTMLCollection动态的,意味着如果文档结构发生变化,它会实时反映这些变化。
    • NodeList (由 querySelectorAll 返回的) 通常是静态的,它是在查询那一刻的快照。
    • HTMLCollectionNodeList 都可以通过索引访问元素,并且有 length 属性,但 NodeList 可以使用 forEach 方法,而 HTMLCollection 通常需要转换成数组 (Array.from()) 后才能使用 forEach
2. 创建新 DOM 元素 ("增")

我们可以动态地创建新的 HTML 元素。

  • document.createElement('tagName'): 创建指定标签名的新元素节点。 
    const newDiv = document.createElement('div');
    const newParagraph = document.createElement('p');
    
  • document.createTextNode('text'): 创建一个包含指定文本的文本节点。 
    const paragraphText = document.createTextNode('这是新段落的内容。');
    
  • document.createDocumentFragment(): 创建一个文档片段。这是一个轻量级的 DOM 容器,当你需要一次性向 DOM 中添加多个元素时,先将它们添加到 DocumentFragment 中,然后再将 DocumentFragment 一次性添加到主 DOM 树,这样可以减少页面重绘和回流,提高性能。 
    const fragment = document.createDocumentFragment();
    const item1 = document.createElement('li');
    item1.textContent = '列表项1';
    const item2 = document.createElement('li');
    item2.textContent = '列表项2';
    
    fragment.appendChild(item1);
    fragment.appendChild(item2);
    // 稍后可以将 fragment 添加到 ul 元素中
    
3. 添加/插入 DOM 元素 ("增")

创建元素后,需要将它们插入到 DOM 树的特定位置。

  • parentNode.appendChild(childNode): 将 childNode 作为 parentNode 的最后一个子节点添加。 
    const parentDiv = document.getElementById('parent');
    const newChild = document.createElement('p');
    newChild.textContent = '我是新来的子节点。';
    parentDiv.appendChild(newChild);
    
  • parentNode.insertBefore(newNode, referenceNode): 在 parentNode 的子节点 referenceNode 之前插入 newNode。如果 referenceNodenull,则其行为类似 appendChild。 
    const parent = document.getElementById('listContainer');
    const newItem = document.createElement('li');
    newItem.textContent = '新列表项 (插入)';
    const firstItem = parent.querySelector('li'); // 假设已有列表项
    if (firstItem) {
      parent.insertBefore(newItem, firstItem);
    } else {
      parent.appendChild(newItem); // 如果没有参考节点,则添加到末尾
    }
    
  • 现代插入方法 (更便捷):
    • element.append(...nodesOrDOMStrings): 在 element 的子节点的末尾插入一个或多个节点或 DOM 字符串。
    • element.prepend(...nodesOrDOMStrings): 在 element 的子节点的开头插入。
    • element.before(...nodesOrDOMStrings): 在 element 之前插入。
    • element.after(...nodesOrDOMStrings): 在 element 之后插入。 
    const container = document.getElementById('container');
    const p1 = document.createElement('p');
    p1.textContent = '段落1 (append)';
    const p2 = document.createElement('p');
    p2.textContent = '段落2 (prepend)';
    
    container.append(p1, '一些文本'); // 可以同时添加节点和文本
    container.prepend(p2);
    
4. 修改 DOM 元素 ("改")

获取或创建元素后,我们可以修改其内容、属性和样式。

  • 修改内容:

    • element.textContent: 获取或设置元素的纯文本内容。它会自动转义 HTML 标签,所以相对安全,性能也较好。 
      const greeting = document.getElementById('greeting');
      greeting.textContent = '你好,世界!'; // 设置文本
      console.log(greeting.textContent); // 获取文本
      
    • element.innerHTML: 获取或设置元素的 HTML 内容。可以用来插入 HTML 结构,但要注意 XSS (跨站脚本) 风险。只应在内容来源可信或已做适当清理时使用。 
      const contentDiv = document.getElementById('content');
      contentDiv.innerHTML = '<strong>加粗文本</strong> 和一个 <a href="#">链接</a>';
      
    • element.innerText: 与 textContent 类似,但它会考虑元素的CSS样式(例如 display: none 的元素内容不会被获取),并且在设置时会触发重排。通常推荐使用 textContent
  • 修改属性 :

    • element.getAttribute('attrName'): 获取指定属性的值。
    • element.setAttribute('attrName', 'value'): 设置或更新指定属性的值。
    • element.hasAttribute('attrName'): 检查元素是否拥有指定属性。
    • element.removeAttribute('attrName'): 移除指定属性。 
      const myImage = document.getElementById('myImage');
      myImage.setAttribute('src', 'new_image.jpg');
      myImage.setAttribute('alt', '一张新图片');
      console.log(myImage.getAttribute('src')); // new_image.jpg
      if (myImage.hasAttribute('data-custom')) {
          myImage.removeAttribute('data-custom');
      }
      
    • 直接属性访问: 对于标准的 HTML 属性,通常可以直接作为元素的 JavaScript 对象属性来访问和修改,如 element.id, element.className, element.src, element.href, element.value (对于表单元素)。 
      const link = document.querySelector('a');
      link.href = 'https://www.example.com';
      link.id = 'mySpecialLink';
      
      属性 (Attributes) vs 特性 (Properties): HTML 标签上写的叫属性 (attribute),JS 对象上的叫特性 (property)。多数情况下它们是同步的,但有些例外 (如 inputvalue attribute 反映初始值,而 value property 反映当前值)。
  • 修改样式 :

    • element.style.property = 'value': 直接修改元素的内联样式。CSS 属性名需要转换为驼峰式命名 (如 background-color 变为 backgroundColor)。 
      const box = document.getElementById('box');
      box.style.width = '200px';
      box.style.height = '200px';
      box.style.backgroundColor = 'lightblue';
      box.style.border = '1px solid blue';
      
    • element.className: 获取或设置元素的 class 属性字符串。设置会覆盖所有现有类名。 
      const message = document.getElementById('message');
      message.className = 'alert success'; // 设置多个类
      
    • element.classList (推荐): 提供更方便的方法来操作类名,返回一个 DOMTokenList
      • add('className1', 'className2', ...): 添加一个或多个类名。
      • remove('className1', 'className2', ...): 移除一个或多个类名。
      • toggle('className', forceBoolean?): 切换类名。如果存在则移除,不存在则添加。可选的 forceBoolean 参数可以强制添加或移除。
      • contains('className'): 检查是否存在指定类名。
      const panel = document.getElementById('infoPanel');
      panel.classList.add('visible', 'highlight');
      panel.classList.remove('hidden');
      if (panel.classList.contains('highlight')) {
        console.log('Panel is highlighted.');
      }
      panel.classList.toggle('expanded');
      
    • getComputedStyle(element).property: 获取元素最终计算后的样式值(包括来自外部CSS文件、内部样式表和内联样式的综合结果)。返回的是只读的。 
      const styledElement = document.getElementById('styledElement');
      const computedColor = window.getComputedStyle(styledElement).color;
      console.log('Computed color:', computedColor);
      
5. 删除 DOM 元素 ("删")
  • parentNode.removeChild(childNode): 从 parentNode 中移除指定的 childNode。被移除的节点仍然存在于内存中,可以被重新添加到文档中。 
    const list = document.getElementById('myList');
    const itemToRemove = document.getElementById('item-2');
    if (list && itemToRemove && itemToRemove.parentNode === list) { // 确保是其子节点
      list.removeChild(itemToRemove);
    }
    
  • 现代方法:element.remove(): 直接从其父节点中移除该元素自身。更简洁。 
    const oldAd = document.getElementById('oldAdBanner');
    if (oldAd) {
      oldAd.remove();
    }
    
6. 替换 DOM 元素 ("改")
  • parentNode.replaceChild(newNode, oldNode): 用 newNode 替换 parentNode 中的子节点 oldNode。 
    const container = document.getElementById('contentArea');
    const oldParagraph = container.querySelector('p.old-text');
    const newParagraph = document.createElement('p');
    newParagraph.textContent = '这是更新后的内容。';
    newParagraph.className = 'new-text';
    
    if (container && oldParagraph) {
      container.replaceChild(newParagraph, oldParagraph);
    }
    

Part 3: JavaScript 事件处理机制

事件是用户与网页交互(如点击按钮、输入文本)或浏览器自身状态改变(如页面加载完成)时发出的信号。JavaScript 允许我们“监听”这些事件,并在事件发生时执行特定的代码。

1. 什么是事件?
  • 用户行为: click (点击), keydown (按下键盘), mousemove (鼠标移动), submit (表单提交) 等。
  • 浏览器行为: load (页面或资源加载完成), DOMContentLoaded (DOM树构建完成), error (资源加载错误) 等。
  • 事件驱动编程: JavaScript 在浏览器中的很多行为都是基于事件驱动的,即代码的执行是由特定事件的发生来触发的。
2. 事件监听器 

我们通过事件监听器来响应事件。

  • HTML 内联属性 (不推荐)

    <button onclick="alert('按钮被点击了!')">点我 (内联)</button>
    

    这种方式将 JavaScript 代码直接写在 HTML 中,不利于代码分离和维护。

  • DOM0 级事件处理:

    将一个函数赋值给元素的事件处理属性 (如 onclick, onmouseover)

    const button0 = document.getElementById('btn0');
    button0.onclick = function() {
      console.log('DOM0 级事件:按钮被点击!');
    };
    // 再次赋值会覆盖之前的
    // button0.onclick = function() { console.log('新的点击处理'); };
    

    缺点是每个事件类型只能绑定一个处理函数。

  • DOM2 级事件处理 (addEventListener - 推荐):

    这是现代 Web 开发中最常用和推荐的方式。

    • 语法: element.addEventListener('eventType', handlerFunction, useCaptureOrOptionsObject)
      • eventType: 事件类型字符串,如 'click', 'mouseover', 'keydown' (不需要 on 前缀)。
      • handlerFunction: 事件发生时要执行的函数。
      • useCaptureOrOptionsObject:
        • 布尔值:true 表示在捕获阶段处理事件,false (默认) 表示在冒泡阶段处理。
        • 对象:可以传递更详细的选项,如 { capture: true, once: true, passive: true }
          • once: true: 处理函数最多执行一次后自动移除。
          • passive: true: 向浏览器表明处理函数不会调用 event.preventDefault(),有助于优化滚动等性能。
    • 移除监听器: element.removeEventListener('eventType', handlerFunction, useCaptureOrBoolean)
      • 重要: 移除时传入的 handlerFunction 必须与添加时传入的函数是同一个引用。匿名函数无法通过这种方式移除。
    const button2 = document.getElementById('btn2');
    
    function handleClick() {
      console.log('DOM2 级事件:按钮被点击!');
    }
    function anotherClickHandler() {
        console.log('DOM2 级事件:按钮也被另一个函数处理了!')
    }
    
    button2.addEventListener('click', handleClick);
    button2.addEventListener('click', anotherClickHandler); // 可以为同一事件添加多个处理函数
    
    // 移除 handleClick
    // button2.removeEventListener('click', handleClick);
    
3. 事件对象 

当事件发生时,浏览器会自动创建一个包含事件相关信息的对象,并将其作为参数传递给事件处理函数。通常我们将其命名为 evente

  • 常用属性:

    • event.type: (字符串) 事件的类型,如 'click'
    • event.target: (元素节点) 实际触发事件的那个元素(事件源)。
    • event.currentTarget: (元素节点) 事件监听器当前附加到的那个元素。在事件冒泡过程中,target 是不变的,而 currentTarget 会是路径上每个正在处理事件的元素。
    • event.bubbles: (布尔值) 事件是否冒泡。
    • event.cancelable: (布尔值) 事件是否可以被取消默认行为。
  • 常用方法:

    • event.preventDefault(): 阻止事件的默认浏览器行为。例如,阻止链接的跳转,阻止表单的提交。
    • event.stopPropagation(): 停止事件在 DOM 树中的进一步传播(冒泡或捕获)。
  • 特定事件类型的属性:

    • 鼠标事件 :
      • event.clientX, event.clientY: 鼠标指针相对于浏览器窗口可视区域的坐标。
      • event.pageX, event.pageY: 鼠标指针相对于整个文档的坐标。
      • event.button: 按下的鼠标按钮 (0: 左键, 1: 中键, 2: 右键)。
      • event.altKey, event.ctrlKey, event.shiftKey, event.metaKey (Mac上的Command键): 布尔值,表示相应的功能键是否被按下。
    • 键盘事件 :
      • event.key: 按下的键的字符值 (如 'a', 'Enter', 'ArrowUp')。推荐使用。
      • event.code: 按下的物理键的代码 (如 'KeyA', 'Enter', 'ArrowUp'),不受键盘布局影响。
      • event.altKey, event.ctrlKey, event.shiftKey, event.metaKey: 同上。
    const myLink = document.getElementById('myLink');
    myLink.addEventListener('click', function(event) {
      event.preventDefault(); // 阻止链接默认的跳转行为
      console.log('链接被点击,但默认行为已阻止。');
      console.log('事件类型:', event.type);           // "click"
      console.log('目标元素:', event.target);         // <a> 元素
      console.log('当前目标:', event.currentTarget); // <a> 元素
    });
    
    const inputField = document.getElementById('textInput');
    inputField.addEventListener('keydown', function(e) {
      console.log(`按下的键: ${e.key}, 物理键码: ${e.code}`);
      if (e.key === 'Enter') {
        console.log('回车键被按下!');
        // e.preventDefault(); // 如果在表单中,可能需要阻止默认提交
      }
    });
    

4. 事件流:捕获与冒泡的奇妙旅程

想象一下,当你在网页上的一个按钮上点击时,这个“点击”的信号并不仅仅是按钮自己知道,它其实经历了一场从“天”到“地”,再从“地”到“天”的旅程。这个旅程就是事件流,它描述了事件在 DOM 树中传播的顺序。

事件流主要包含三个阶段:

  1. 捕获阶段 :圣旨下达

    • 当事件发生时,它首先像一道“圣旨”从最高层(window对象)开始,沿着 DOM 树的层级关系,由外向内,一层层向下传播。它会经过目标元素的祖先元素(比如包含按钮的 div,再往上的 body 等),一直“下达”到实际触发事件的那个元素(我们称之为“目标元素”,比如你点击的那个按钮 event.target)。
    • 在这个阶段,如果你给沿途的父元素设置了“在捕获阶段就处理”的监听器,它们就能提前“截获”并处理这个事件,就像大臣们在圣旨到达最终目的地前就能阅览一样。
  2. 目标阶段 :正主响应

    • “圣旨”(事件)终于到达了它的最终目的地——目标元素(比如那个被点击的按钮)。
    • 浏览器会检查这个目标元素本身是否注册了针对该事件的监听器。如果注册了,就会执行相应的处理函数。此时,无论是设置为捕获阶段还是冒泡阶段(默认)的监听器,只要是绑定在目标元素上的,都会在这个阶段被触发(具体顺序取决于绑定时的设置和浏览器实现,但通常捕获优先于冒泡)。
  3. 冒泡阶段:消息上报

    • 在目标元素处理完事件(或者即使没有处理)之后,这个事件通常不会就此消失。它会像水中的气泡一样,从目标元素开始,沿着 DOM 树的层级关系,由内向外,一层层向上“冒泡”传播。它会再次经过目标元素的父元素、祖父元素,一直到文档的根节点,最终回到 window 对象。
    • 这是大多数事件(如 click, mouseover 等)的默认行为。如果你给沿途的父元素设置了“在冒泡阶段处理”(这是 addEventListener 的默认情况)的监听器,它们就能在这个事件“回程”的阶段响应事件,就像地方官员逐级向上传达消息一样。

如何在代码中控制?

addEventListener 方法的第三个参数用来精确控制监听器在哪个阶段触发:

  • element.addEventListener('click', myFunction, true); 将第三个参数设置为 true,则 myFunction 会在捕获阶段执行。
  • element.addEventListener('click', myFunction, false); (或者不写第三个参数,因为默认值就是 false),则 myFunction 会在冒泡阶段执行。

代码示例:

<div id="grandparent" style="padding: 30px; background-color: lightblue;">
  祖父
  <div id="parent" style="padding: 20px; background-color: lightgreen;">
    父亲
    <button id="child" style="padding: 10px; background-color: yellow;">孩子 </button>
  </div>
</div>
const grandparent = document.getElementById('grandparent');
const parent = document.getElementById('parent');
const child = document.getElementById('child');

// --- 捕获阶段监听 ---
grandparent.addEventListener('click', function(event) {
  console.log(`捕获阶段:监听器在 ${this.id},事件目标是 ${event.target.id}`);
}, true);

parent.addEventListener('click', function(event) {
  console.log(`捕获阶段:监听器在 ${this.id},事件目标是 ${event.target.id}`);
}, true);

child.addEventListener('click', function(event) {
  // 对于目标元素,捕获阶段的监听器会在目标阶段早期被触发
  console.log(`捕获/目标阶段:监听器在 ${this.id},事件目标是 ${event.target.id}`);
}, true);


// --- 冒泡阶段监听 (默认) ---
grandparent.addEventListener('click', function(event) {
  console.log(`冒泡阶段:监听器在 ${this.id},事件目标是 ${event.target.id}`);
});

parent.addEventListener('click', function(event) {
  console.log(`冒泡阶段:监听器在 ${this.id},事件目标是 ${event.target.id}`);
  // 如果在这里调用 event.stopPropagation(); 下面的 grandparent 的冒泡监听就不会执行了
  // event.stopPropagation();
});

child.addEventListener('click', function(event) {
  // 对于目标元素,冒泡阶段的监听器会在目标阶段晚期(或紧随其后)被触发
  console.log(`冒泡/目标阶段:监听器在 ${this.id},事件目标是 ${event.target.id}`);
});

console.log("--- 请点击 '孩子' 按钮 ---");

当你点击 "孩子 " 按钮后,控制台大致的输出顺序会清晰地展示事件流的路径——先从外向内(捕获),到达目标,再从内向外(冒泡):

  1. 捕获阶段:监听器在 grandparent,事件目标是 child
  2. 捕获阶段:监听器在 parent,事件目标是 child
  3. 捕获/目标阶段:监听器在 child,事件目标是 child
  4. 冒泡/目标阶段:监听器在 child,事件目标是 child
  5. 冒泡阶段:监听器在 parent,事件目标是 child
  6. 冒泡阶段:监听器在 grandparent,事件目标是 child

阻止事件传播:event.stopPropagaton()

如果你不希望事件在当前元素处理后,继续向上冒泡(或者在捕获阶段继续向下传播),可以在事件处理函数中调用 event.stopPropagation()。这就好比在消息传递的某个环节说:“消息到此为止,不用再往其他地方传了!”这在某些情况下可以避免父元素上不必要的事件处理。


5. 事件委托/代理:聪明的事件管理员

想象一个场景:你有一个很长的待办事项列表 (<ul>),列表中的每一项 (<li>) 点击后都需要标记为完成。如果列表有100项,难道我们要给这100个 <li> 元素都单独绑定一个点击事件监听器吗?

这样做会有两个主要问题:

  1. 性能开销: 绑定大量的事件监听器会占用更多的内存和CPU资源,尤其是在元素非常多的时候。
  2. 动态元素的烦恼: 如果你通过JavaScript动态地向列表中添加了新的待办事项,新添加的这些 <li> 元素是不会自动拥有之前绑定的点击事件的,你还得手动为它们再绑定一次,非常麻烦。

事件委托 就是来解决这些问题的“聪明管理员”!

核心思想:

与其给每个子元素都派一个“警卫”(事件监听器),不如只在它们的共同父元素(比如 <ul>)上派一个“总警卫”。利用事件冒泡的原理(还记得吗?事件会从目标元素向上传播),当任何一个子元素(比如某个 <li>)被点击时,这个点击事件会“冒泡”到父元素 <ul> 上。这时,父元素上的“总警卫”就会被触发。

工作原理:

  1. 将事件监听器绑定在这些子元素的共同父元素上。
  2. 当子元素上的事件(如 click)发生并冒泡到父元素时,父元素的监听器被触发。
  3. 在父元素的监听器函数内部,我们可以通过事件对象 event.target 来判断究竟是哪个子元素真正触发了这个事件。event.target 指向的是实际被点击的那个最具体的元素(也就是我们的“消息来源”)。

优点:

  • 高效: 只需要一个事件监听器,大大减少了内存占用和初始设置的开销。
  • 灵活: 对于动态添加的子元素(比如用户新加的待办事项),无需重新绑定事件,它们产生的事件同样会冒泡到父元素,被统一处理,非常方便。
  • 简洁: 代码管理更方便,逻辑更集中。

代码示例:

<ul id="todoList" style="border:1px solid #ccc; padding: 10px;">
  <li>学习 JavaScript</li>
  <li>做个练习项目</li>
  <li>喝杯水</li>
</ul>
<button id="addTodoBtn">添加待办事项</button>
<style>
  .completed {
    text-decoration: line-through;
    color: grey;
  }
</style>
const todoList = document.getElementById('todoList');
const addTodoBtn = document.getElementById('addTodoBtn');

// 在父元素 ul 上设置事件监听器
todoList.addEventListener('click', function(event) {
  // event.target 是用户实际点击的那个元素
  // 我们只关心被点击的是否是 <li> 元素
  if (event.target && event.target.tagName === 'LI') {
    // event.target.tagName 返回的是大写的标签名,如 'LI'
    // 也可以用更现代的方式 event.target.matches('li')

    const listItemText = event.target.textContent;
    console.log(`你点击了待办事项: "${listItemText}"`);

    // 给被点击的 <li> 添加/移除一个 "completed" 样式类
    event.target.classList.toggle('completed'); // toggle 会在有和无之间切换
    // 你也可以在这里做其他操作,比如删除、编辑等
  }
});

// 动态添加新的待办事项
let todoCounter = 3;
addTodoBtn.addEventListener('click', function() {
  todoCounter++;
  const newTodo = document.createElement('li');
  newTodo.textContent = `新的待办 ${todoCounter}`;
  todoList.appendChild(newTodo);
  // 注意:新添加的 <li> 无需单独绑定事件,点击它时,父元素 ul 的监听器依然会处理!
});

在这个例子中,无论列表有多少项,或者之后动态添加了多少项,我们都只需要 todoList (父元素 <ul>) 上的那一个点击事件监听器。它就像一个聪明的管理员,有效地管理着所有子项的点击事件。


6. 常见事件类型
  • 鼠标事件:
    • click: 单击。
    • dblclick: 双击。
    • mousedown: 鼠标按钮按下。
    • mouseup: 鼠标按钮释放。
    • mousemove: 鼠标在元素上移动。
    • mouseover: 鼠标指针进入元素或其子元素。
    • mouseout: 鼠标指针离开元素或进入其子元素。
    • mouseenter: 鼠标指针进入元素 (不冒泡,不关心子元素)。
    • mouseleave: 鼠标指针离开元素 (不冒泡,不关心子元素)。
  • 键盘事件:
    • keydown: 按下键盘键。
    • keyup: 释放键盘键。
    • keypress: (已不推荐使用) 按下产生字符的键时触发。
  • 表单事件:
    • submit: 表单提交时 (通常在 <form> 元素上监听)。
    • change: 表单元素内容改变且失去焦点后 (如 <input type="text">, <select>, <input type="checkbox">)。
    • input: 表单元素内容实时改变时 (如 <input type="text">, <textarea>)。
    • focus: 元素获得焦点。
    • blur: 元素失去焦点。
  • 窗口/文档事件:
    • load: 整个页面及所有资源 (图片、CSS等) 加载完成后在 window 上触发。也用于图片等单个资源的加载完成。
    • DOMContentLoaded: HTML文档被完全加载和解析完成之后触发,无需等待样式表、图像和子框架的完成加载 (在 document 上触发)。通常比 load 事件更早触发,是执行初始化脚本的好时机。
    • resize: 浏览器窗口大小改变时在 window 上触发。
    • scroll: 页面或元素滚动时触发。
  • 触摸事件 (主要用于移动设备):
    • touchstart, touchmove, touchend, touchcancel

Part 4: 实践与最佳做法:写出高效、优雅的 DOM 与事件代码

掌握了 DOM 操作和事件处理的基础后,如何写出更专业、性能更好、更易于维护的代码呢?以下是一些重要的实践和最佳做法,并配有代码示例帮助理解。

1. 提升性能的考量 

频繁或不当的 DOM 操作是导致页面性能下降的常见原因。

  • a. 减少 DOM 操作次数:批量处理是王道

    不良示例: 在循环中逐个添加元素到 DOM。

    // 假设有一个数组 itemsData = ['苹果', '香蕉', '橙子'];
    // const myList = document.getElementById('myList');
    
    // itemsData.forEach(itemText => {
    //   const listItem = document.createElement('li');
    //   listItem.textContent = itemText;
    //   myList.appendChild(listItem); // 每次循环都操作DOM,导致多次重绘/回流
    // });
    // console.log("不推荐:逐个添加列表项");
    

    推荐示例: 使用 DocumentFragment 进行批量添加。DocumentFragment 是一个轻量级的 DOM 容器,你可以先把所有新元素添加到它里面,最后再一次性把 DocumentFragment 添加到主 DOM 树,这样只会触发一次重绘/回流。

    const itemsData = ['苹果', '香蕉', '橙子'];
    const myList = document.getElementById('myList'); // 假设页面有 <ul id="myList"></ul>
    
    if (myList) {
        const fragment = document.createDocumentFragment(); // 创建文档片段
    
        itemsData.forEach(itemText => {
          const listItem = document.createElement('li');
          listItem.textContent = itemText;
          fragment.appendChild(listItem); // 先添加到 fragment
        });
    
        myList.appendChild(fragment); // 一次性添加到真实DOM
        console.log("推荐:使用 DocumentFragment 批量添加列表项");
    }
    
  • b. 避免布局抖动

    当你在 JavaScript 中连续地读取元素的布局属性(如 offsetHeight, offsetTop, clientWidth 等),然后又立即修改这些可能影响布局的属性,接着又去读取,浏览器可能需要在每次读/写之间强制重新计算布局,这非常耗性能。

    不良示例: 循环中交替读写布局属性。

    // const elements = document.querySelectorAll('.box-item');
    // // 假设我们想让每个盒子的宽度等于其高度
    // elements.forEach(el => {
    //   // 读取 offsetHeight (导致回流以获取精确值)
    //   const h = el.offsetHeight;
    //   // 修改 width (可能导致重绘/回流)
    //   el.style.width = h + 'px'; // 每次循环都可能触发强制同步布局
    // });
    // console.log("不推荐:循环中交替读写布局属性");
    

    推荐示例: 先集中读取,再集中写入。

    const elements = document.querySelectorAll('.box-item'); // 假设页面有一些 .box-item 元素
    const heights = [];
    
    // 步骤1: 集中读取所有需要的值
    elements.forEach(el => {
      heights.push(el.offsetHeight);
    });
    
    // 步骤2: 集中写入(修改DOM)
    elements.forEach((el, index) => {
      el.style.width = heights[index] + 'px';
    });
    console.log("推荐:分离读写操作,避免布局抖动");
    
  • c. 优先使用事件委托 

    当有大量相似元素需要相同的事件处理时(比如一个长列表的每一项),为每个元素都绑定监听器不如在它们的共同父元素上设置一个监听器。这在 Part 3 的“事件委托”部分已有详细代码演示。

    核心优势回顾: 减少内存占用,动态添加的子元素也能自动应用事件处理。

  • d. 合理使用节流与防抖 

    对于高频触发的事件,如 scroll (滚动)、resize (窗口大小调整)、mousemove (鼠标移动) 以及输入框的 input 事件,如果不加控制,事件处理函数会被过度执行,严重影响性能。

    • 防抖 : 任务频繁触发的情况下,只有当任务触发的间隔超过指定时间时,任务才会执行一次。适用于用户停止操作后才需要响应的场景(如输入框搜索建议)。
    • 节流: 规定在一个单位时间内,只能触发一次函数。如果这个单位时间内触发多次函数,只有一次生效。适用于需要规律性响应的场景(如滚动加载)。

    代码演示(概念性用法,假设 debouncethrottle 函数已定义):

    // 假设已在别处定义了 debounce 和 throttle 函数
    // function debounce(func, delay) { /* ... */ }
    // function throttle(func, limit) { /* ... */ }
    
    // const searchInput = document.getElementById('searchInput');
    // if (searchInput && typeof debounce === 'function') {
    //   searchInput.addEventListener('input', debounce(function(event) {
    //     console.log('防抖搜索:', event.target.value);
    //     // 实际的搜索逻辑
    //   }, 500)); // 用户停止输入500ms后执行
    // }
    
    // if (typeof throttle === 'function') {
    //   window.addEventListener('scroll', throttle(function() {
    //     console.log('节流处理滚动事件');
    //     // 实际的滚动处理逻辑,比如判断是否加载更多
    //   }, 200)); // 每200ms最多执行一次
    // }
    console.log("提示:对于高频事件,务必使用节流或防抖优化。");
    

    (注:完整的 节流和防抖实现较为复杂,这里仅演示调用方式。)

2. 提升代码组织与可读性 
  • a. 关注点分离:结构、样式、行为各司其职

    尽量将 HTML (结构)、CSS (样式) 和 JavaScript (行为) 分离开。

    不良示例 (混合):

  • <!DOCTYPE html>
    <html>
    <head>
        <title>混合示例</title>
    </head>
    <body>
    
        <button id="mixedBtn"
                style="color: red; font-weight: bold; padding: 10px;"
                onclick="this.textContent = '我被点击了!'; console.log('混合按钮被点击'); alert('弹窗!');">
            点我看看 (样式和行为都在HTML里)
        </button>
    
    </body>
    </html>

    推荐示例 (分离):

    <button id="separatedBtn">点我</button>
    <style> /* 通常CSS在外部文件 */
      #separatedBtn { color: blue; }
      #separatedBtn.clicked { font-weight: bold; }
    </style>
    
    const separatedBtn = document.getElementById('separatedBtn');
    if (separatedBtn) {
      separatedBtn.addEventListener('click', function() {
        console.log('按钮被点击!');
        this.textContent = '已点击';
        this.classList.add('clicked');
      });
      console.log("推荐:HTML, CSS, JS 分离,关注点清晰。");
    }
    
  • b. 清晰、有意义的命名

    为变量、函数(尤其是事件处理函数)使用能够清晰表达其用途的名称。

    不够清晰的命名:

    // const b = document.getElementById('btn1');
    // function doIt(e) {
    //   // ...
    // }
    // b.addEventListener('click', doIt);
    

    更清晰的命名:

    const submitFormButton = document.getElementById('userRegistrationSubmitBtn');
    
    function handleUserRegistrationSubmit(event) {
      event.preventDefault(); // 假设是表单提交
      console.log('处理用户注册表单提交...');
      // ... 表单验证和提交逻辑 ...
    }
    
    // if (submitFormButton) {
    //   submitFormButton.addEventListener('click', handleUserRegistrationSubmit);
    // }
    console.log("推荐:使用清晰的变量和函数命名。");
    
  • c. 及时移除不再需要的事件监听器

    当元素从 DOM 中被移除,或者在单页应用 (SPA) 中组件被销毁时,如果其上绑定的事件监听器没有被移除,可能会导致内存泄漏或意外行为。

    代码演示(移除自身监听器并移除元素):

    const ephemeralButton = document.createElement('button');
    ephemeralButton.textContent = '点我后消失 (并移除监听器)';
    document.body.appendChild(ephemeralButton); // 先添加到body才能看到
    
    function handleClickAndSelfDestruct() {
      console.log('临时按钮被点击!');
      // 关键:在元素移除前,先移除其上的事件监听器
      ephemeralButton.removeEventListener('click', handleClickAndSelfDestruct);
      ephemeralButton.remove(); // 从DOM中移除按钮
      console.log('临时按钮及其监听器已移除。');
    }
    
    ephemeralButton.addEventListener('click', handleClickAndSelfDestruct);
    

    在复杂的SPA框架中,通常框架本身会提供组件生命周期钩子函数,让你在组件卸载时执行清理操作,包括移除事件监听器。

3. 关注可访问性 (Accessibility - a11y)

确保你通过 JavaScript 创建和控制的交互对所有用户(包括使用辅助技术的用户)都是可访问的。

  • a. 确保键盘可操作性

    如果使用非标准 HTML 元素(如 <div>)来模拟按钮或其他交互控件,需要确保它们可以通过键盘聚焦和操作。

    代码演示(使 <div> 像按钮一样可操作):

    const customControl = document.createElement('div');
    customControl.textContent = '自定义可操作控件';
    customControl.setAttribute('role', 'button'); // 1. 语义化:告诉辅助技术这是一个按钮
    customControl.setAttribute('tabindex', '0');  // 2. 使其可通过 Tab键 聚焦
    customControl.style.cssText = 'padding: 8px; border: 1px solid grey; display: inline-block; cursor: pointer;'; // 简单样式
    
    function handleCustomControlAction() {
      console.log('自定义控件被激活!');
      alert('自定义控件激活成功!');
    }
    
    customControl.addEventListener('click', handleCustomControlAction);
    customControl.addEventListener('keydown', function(event) {
      // 3. 允许通过 Enter 或 Space 键激活
      if (event.key === 'Enter' || event.key === ' ') { // 或者 event.code === 'Space'
        event.preventDefault(); // 防止空格键滚动页面等默认行为
        handleCustomControlAction();
      }
    });
    
    document.body.appendChild(customControl);
    console.log("提示:自定义交互元素需确保键盘可访问性。");
    
  • b. 使用 ARIA 属性增强语义

    对于通过 DOM 操作创建的复杂或非标准交互组件(如自定义下拉菜单、模态框、选项卡面板等),HTML 原生标签可能无法完全表达其角色、状态和属性。此时,应使用 ARIA (Accessible Rich Internet Applications) 属性来补充这些语义信息。

    代码演示(简单的自定义开关状态):

    const myToggleSwitch = document.createElement('div');
    myToggleSwitch.setAttribute('role', 'switch');    // 角色:这是一个开关
    myToggleSwitch.setAttribute('aria-checked', 'false'); // 初始状态:未选中
    myToggleSwitch.setAttribute('tabindex', '0');       // 可聚焦
    myToggleSwitch.textContent = '关';
    myToggleSwitch.style.cssText = 'padding: 5px; border: 1px solid black; display: inline-block; cursor: pointer; user-select: none;';
    
    let isSwitchOn = false;
    function toggleTheSwitch() {
      isSwitchOn = !isSwitchOn;
      myToggleSwitch.setAttribute('aria-checked', isSwitchOn.toString());
      myToggleSwitch.textContent = isSwitchOn ? '开' : '关';
      myToggleSwitch.style.backgroundColor = isSwitchOn ? 'lightgreen' : 'lightcoral';
      console.log(`开关状态: ${isSwitchOn ? '开' : '关'}`);
    }
    
    myToggleSwitch.addEventListener('click', toggleTheSwitch);
    myToggleSwitch.addEventListener('keydown', (event) => {
      if (event.key === 'Enter' || event.key === ' ') {
        event.preventDefault();
        toggleTheSwitch();
      }
    });
    
    document.body.appendChild(myToggleSwitch);
    console.log("提示:复杂自定义组件应使用ARIA属性增强语义。");
    

    ARIA 是一个相对深入的主题,但了解其基本作用和在必要时查阅相关文档非常重要。

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

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

相关文章

Java EE初阶——初识多线程

1. 认识线程 线程是操作系统能够进行运算调度的最小单位。它被包含在进程之中&#xff0c;是进程中的实际运作单位。 基本概念&#xff1a;一个进程可以包含多个线程&#xff0c;这些线程共享进程的资源&#xff0c;如内存空间、文件描述符等&#xff0c;但每个线程都有自己独…

如何删除网上下载的资源后面的文字

这是我在爱给网上下载的音效资源&#xff0c;但是发现资源后面跟了一大段无关紧要的文本&#xff0c;但是修改资源名称后还是有。解决办法是打开属性然后删掉资源的标签即可。

FPGA图像处理(5)------ 图片水平镜像

利用bram形成双缓冲&#xff0c;如下图配置所示&#xff1a; wr_flag 表明 buffer0写 还是 buffer1写 rd_flag 表明 buffer0读 还是 buffer1读 通过写入逻辑控制(结合wr_finish) 写哪个buffer &#xff1b;写地址 进而控制ip的写使能 通过状态缓存来跳转buffer的…

day21python打卡

知识点回顾&#xff1a; LDA线性判别PCA主成分分析t-sne降维 还有一些其他的降维方式&#xff0c;也就是最重要的词向量的加工&#xff0c;我们未来再说 作业&#xff1a; 自由作业&#xff1a;探索下什么时候用到降维&#xff1f;降维的主要应用&#xff1f;或者让ai给你出题&…

ERP学习(一): 用友u8安装

安装&#xff1a; https://www.bilibili.com/video/BV1Pp4y187ot/?spm_id_from333.337.search-card.all.click&vd_sourced514093d85ee628d1f12310b13b1e59b 我个人用vmware16&#xff0c;这位up已经把用友软件和环境&#xff08;sqlserver2008&#xff09; 都封城vmx文件了…

01 | 大模型微调 | 从0学习到实战微调 | AI发展与模型技术介绍

一、导读 作为非AI专业技术开发者&#xff08;我是小小爬虫开发工程师&#x1f60b;&#xff09; 本系列文章将围绕《大模型微调》进行学习&#xff08;也是我个人学习的笔记&#xff0c;所以会持续更新&#xff09;&#xff0c;最后以上手实操模型微调的目的。 (本文如若有…

海康相机无损压缩

设置无损压缩得到更高的带宽和帧率&#xff01;

从机器人到调度平台:超低延迟RTMP|RTSP播放器系统级部署之道

✅ 一、模块定位&#xff1a;跨平台、超低延迟、系统级稳定的音视频直播播放器内核 在无人机、机器人、远程操控手柄等场景中&#xff0c;低延迟的 RTSP/RTMP 播放器并不是“可有可无的体验优化”&#xff0c;而是系统能否闭环、操控是否安全的关键组成。 Windows和安卓播放RT…

研发效率破局之道阅读总结(5)管理文化

研发效率破局之道阅读总结(5)管理文化 Author: Once Day Date: 2025年5月10日 一位热衷于Linux学习和开发的菜鸟&#xff0c;试图谱写一场冒险之旅&#xff0c;也许终点只是一场白日梦… 漫漫长路&#xff0c;有人对你微笑过嘛… 全系列文章可参考专栏: 程序的艺术_Once-Day…

单因子实验 方差分析

本文是实验设计与分析&#xff08;第6版&#xff0c;Montgomery著傅珏生译)第3章单因子实验 方差分析python解决方案。本文尽量避免重复书中的理论&#xff0c;着于提供python解决方案&#xff0c;并与原书的运算结果进行对比。您可以从 下载实验设计与分析&#xff08;第6版&a…

Bitacora:基因组组件中基因家族识别和注释的综合工具

软件教程 | Bitacora&#xff1a;基因组组件中基因家族识别和注释的综合工具 https://zhangzl96.github.io/tags#生物信息工具) &#x1f4c5; 官方地址&#xff1a;https://github.com/molevol-ub/bitacora &#x1f52c; 教程版本&#xff1a;BITACORA 1.4 &#x1f4cb; …

【WebRTC-13】是在哪,什么时候,创建编解码器?

Android-RTC系列软重启&#xff0c;改变以往细读源代码的方式 改为 带上实际问题分析代码。增加实用性&#xff0c;方便形成肌肉记忆。同时不分种类、不分难易程度&#xff0c;在线征集问题切入点。 问题&#xff1a;编解码器的关键实体类是什么&#xff1f;在哪里&什么时候…

青少年编程与数学 02-019 Rust 编程基础 01课题、环境准备

青少年编程与数学 02-019 Rust 编程基础 01课题、环境准备 一、Rust核心特性应用场景开发工具社区与生态 二、Rust 和 Python 比较1. **内存安全与并发编程**2. **性能**3. **零成本抽象**4. **跨平台支持**5. **社区与生态系统**6. **错误处理**7. **安全性**适用场景总结 三、…

Redis持久化存储介质评估:NFS与Ceph的适用性分析

#作者&#xff1a;朱雷 文章目录 一、背景二、Redis持久化的必要性与影响1. 持久化的必要性2. 性能与稳定性问题 三、NFS作为持久化存储介质的问题1. 性能瓶颈2. 数据一致性问题3. 存储服务单点故障4. 高延迟影响持久化效率.5. 吞吐量瓶颈 四、Ceph作为持久化存储介质的问题1.…

Ceph 原理与集群配置

一、Ceph 工作原理 1.1.为什么学习 Ceph&#xff1f; 在学习了 NFS 存储之后&#xff0c;我们仍然需要学习 Ceph 存储。这主要是因为不同的存储系统适用于不同的场景&#xff0c;NFS 虽然有其适用之处&#xff0c;但也存在一定的局限性。而 Ceph 能够满足现代分布式、大规模、…

天线的PCB设计

目录 天线模块设计的重要性 天线模块的PCB设计 天线模块设计的重要性 当智能手表突然断连、无人机信号飘忽不定——你可能正在经历一场来自天线模块的"无声抗议"。这个隐藏在电子设备深处的关键组件&#xff0c;就像数字世界的隐形信使&#xff0c;用毫米级的精密结…

C++笔记-set和map的使用(包含multiset和multimap的讲解)

1.序列式容器和关联式容器 前面我们已经接触过STL中的部分容器如:string、vector、list、deque、array、forward_list等&#xff0c;这些容器统称为序列式容器&#xff0c;因为逻辑结构为线性序列的数据结构&#xff0c;两个位置存储的值之间一般没有紧密的关联关系&#xff0…

Linux `ifconfig` 指令深度解析与替代方案指南

Linux `ifconfig` 指令深度解析与替代方案指南 一、核心功能与现状1. 基础作用2. 版本适配二、基础语法与常用操作1. 标准语法2. 常用操作速查显示所有接口信息启用/禁用接口配置IPv4地址修改MAC地址(临时)三、高级配置技巧1. 虚拟接口创建2. MTU调整3. 多播配置4. ARP控制四…

Python pandas 向excel追加数据,不覆盖之前的数据

最近突然看了一下pandas向excel追加数据的方法&#xff0c;发现有很多人出了一些馊主意&#xff1b; 比如用concat,append等方法&#xff0c;这种方法的会先将旧数据df_1读取到内存&#xff0c;再把新数据df_2与旧的合并&#xff0c;形成df_new,再覆盖写入&#xff0c;消耗和速…

【金仓数据库征文】政府项目数据库迁移:从MySQL 5.7到KingbaseES的蜕变之路

摘要&#xff1a;本文详细阐述了政府项目中将 MySQL 5.7 数据库迁移至 KingbaseES 的全过程&#xff0c;涵盖迁移前的环境评估、数据梳理和工具准备&#xff0c;迁移实战中的数据源与目标库连接配置、迁移任务详细设定、执行迁移与过程监控&#xff0c;以及迁移后的质量验证、系…