JavaScript DOM 节点操作

news2025/4/11 1:54:32

目录

一、DOM 节点

节点类型(Node Types)

 二、查找节点

1.查找父节点

1. parentNode

2. parentElement

2.查找子节点

1. childNodes

2. children

3. firstChild / lastChild

4. firstElementChild / lastElementChild

3.查找兄弟节点

1. previousSibling / nextSibling

2. previousElementSibling / nextElementSibling

4.查找特定祖先节点

closest(selector)

5.检查节点关系

contains(node)

6.综合示例

HTML 结构:

JavaScript 操作:

7.注意事项

三、增加节点

1. 创建节点

2. 插入节点

3. 插入 HTML 字符串

四、删除节点

1. 传统方法

2. 现代方法(ES6+)

五、替换节点

1. replaceChild()

2. replaceWith()(ES6+)

六、克隆节点

方法语法

示例代码

克隆注意事项

七、最佳实践与性能优化

示例:


一、DOM 节点

DOM(Document Object Model)节点 是 HTML 或 XML 文档中每个独立单元的抽象表示。整个文档被建模为一个树形结构(DOM 树),其中每个节点对应文档的一部分(如元素、属性、文本等)。节点是 DOM 操作的基本单位,通过 JavaScript 可以动态访问和修改节点,实现页面交互。


节点类型(Node Types)

DOM 规范定义了多种节点类型,每个类型通过 nodeType 属性标识(值为整数常量):

节点类型nodeType 值说明
元素节点1 (Node.ELEMENT_NODE)表示 HTML 标签(如 <div><p>
属性节点2 (Node.ATTRIBUTE_NODE)表示元素的属性(如 id="content",已弃用,推荐用 element.attributes
文本节点3 (Node.TEXT_NODE)表示元素内的文本内容(包括空格和换行)
注释节点8 (Node.COMMENT_NODE)表示 HTML 注释(如 <!-- 注释 -->
文档节点9 (Node.DOCUMENT_NODE)表示整个文档(根节点,如 document
文档片段节点11 (Node.DOCUMENT_FRAGMENT_NODE)轻量级容器,用于批量操作节点(如 document.createDocumentFragment()

 二、查找节点

1.查找父节点

1. parentNode
  • 说明:返回任何类型的父节点(元素、文档、文档片段等)。

  • 示例

    const child = document.querySelector('.child');
    const parent = child.parentNode; // 父元素节点
2. parentElement
  • 说明:仅返回元素类型的父节点。

  • 示例

    const parent = child.parentElement; // 父元素节点(非元素返回 null)

2.查找子节点

1. childNodes
  • 说明:返回所有子节点的动态集合(包括元素、文本、注释等)。

  • 示例

    const children = parent.childNodes; // NodeList
2. children
  • 说明:返回所有元素子节点的动态集合(HTMLCollection)。

  • 示例

    const elements = parent.children; // 仅元素节点
3. firstChild / lastChild
  • 说明:返回第一个/最后一个子节点(包括非元素节点)。

  • 示例

    const first = parent.firstChild; // 可能是文本节点
    const last = parent.lastChild;
4. firstElementChild / lastElementChild
  • 说明:返回第一个/最后一个元素子节点。

  • 示例

    const firstElement = parent.firstElementChild; // 第一个元素子节点
    const lastElement = parent.lastElementChild;

3.查找兄弟节点

1. previousSibling / nextSibling
  • 说明:返回前一个/后一个兄弟节点(包括非元素节点)。

  • 示例

    const prevSibling = element.previousSibling; // 可能是文本节点
    const nextSibling = element.nextSibling;
2. previousElementSibling / nextElementSibling
  • 说明:返回前一个/后一个元素兄弟节点。

  • 示例

    const prevElement = element.previousElementSibling; // 前一个元素节点
    const nextElement = element.nextElementSibling;

4.查找特定祖先节点

closest(selector)
  • 说明:返回匹配选择器的最近祖先元素。

  • 示例

    const ancestor = element.closest('.container'); // 最近的 .container 祖先

5.检查节点关系

contains(node)
  • 说明:检查当前节点是否包含指定后代节点。

  • 示例

    const isChild = parent.contains(child); // true/false

6.综合示例

HTML 结构:
<div id="container">
  <!-- 注释 -->
  <p class="text">段落1</p>
  文本节点
  <p class="text">段落2</p>
</div>
JavaScript 操作:
const container = document.getElementById('container');
const secondP = document.querySelectorAll('.text')[1];

// 查找父节点
console.log(secondP.parentNode.id); // "container"

// 查找子节点
console.log(container.children.length); // 2(两个 p 元素)
console.log(container.firstElementChild.textContent); // "段落1"

// 查找兄弟节点
const firstP = secondP.previousElementSibling;
console.log(firstP.textContent); // "段落1"

// 检查包含关系
console.log(container.contains(secondP)); // true

// 查找特定祖先
const closestDiv = secondP.closest('div');
console.log(closestDiv.id); // "container"

7.注意事项

  1. 动态集合与静态集合

    • childNodes 和 children 是动态集合(HTMLCollection/NodeList),会随 DOM 变化自动更新。

    • querySelectorAll 返回静态 NodeList,不会自动更新。

  2. 空值处理

    • 若节点不存在(如 parentNode 为 null),使用可选链操作符避免报错:

      const parentId = element.parentNode?.id; // 安全访问
  3. 节点类型判断

    • 非元素节点(如文本、注释)无法使用 children 或 firstElementChild 等方法。

  4. 兼容性

    • closest 和 previousElementSibling 等方法在现代浏览器中支持良好,但需注意旧版浏览器兼容性(如 IE)。


三、增加节点

1. 创建节点
// 创建元素节点
const newDiv = document.createElement("div");

// 创建文本节点
const textNode = document.createTextNode("Hello World");
2. 插入节点
方法说明示例
appendChild()在父元素末尾插入子节点parent.appendChild(newDiv)
insertBefore()在指定子节点前插入新节点parent.insertBefore(newDiv, referenceNode)
现代方法append()prepend()before()after()(ES6+)parent.append("追加内容")

示例

// 插入到父元素末尾
const parent = document.getElementById("parent");
parent.appendChild(newDiv);

// 插入到某个子节点前
const referenceNode = document.querySelector(".reference");
parent.insertBefore(newDiv, referenceNode);

// 使用 append 插入多个元素
parent.append(newDiv, "文本节点", document.createElement("span"));
3. 插入 HTML 字符串

使用 insertAdjacentHTML 快速插入:

// 语法:insertAdjacentHTML(position, html)
const container = document.querySelector(".container");
container.insertAdjacentHTML("beforeend", "<div>新内容</div>");

位置参数

  • beforebegin:元素之前

  • afterbegin:元素内部开头

  • beforeend:元素内部末尾

  • afterend:元素之后


四、删除节点

1. 传统方法
// 父元素删除子节点
const parent = document.getElementById("parent");
const child = document.getElementById("child");
parent.removeChild(child); // 需先获取父元素
2. 现代方法(ES6+)
// 直接删除自身
const element = document.querySelector(".to-remove");
element.remove(); // 无需父元素参与

注意

  • 删除后的节点仍存在于内存中,可重新插入 DOM。

  • 若需彻底释放内存,将引用设为 nullelement = null


五、替换节点

1. replaceChild()
const oldNode = document.querySelector(".old");
const newNode = document.createElement("div");
parent.replaceChild(newNode, oldNode); // 用 newNode 替换 oldNode
2. replaceWith()(ES6+)
const oldElement = document.querySelector(".old");
const newElement = document.createElement("div");
oldElement.replaceWith(newElement); // 直接替换自身

六、克隆节点

通过 cloneNode() 方法可以复制节点,常用于动态添加相似结构的元素。

方法语法
const clonedNode = originalNode.cloneNode(deep);
  • deep(可选,默认 false):

    • true:深度克隆,复制节点及其所有子节点。

    • false:浅克隆,仅复制节点本身,不包含子节点。

示例代码
const list = document.getElementById('list');

// 浅克隆:仅复制 <ul> 节点,不含子节点 <li>
const shallowClone = list.cloneNode(false); 

// 深克隆:复制 <ul> 及其所有子节点 <li>
const deepClone = list.cloneNode(true); 

document.body.appendChild(deepClone); // 页面添加克隆的列表
克隆注意事项
  1. ID 重复:克隆的元素的 id 属性与原元素相同,需手动修改以避免冲突。

  2. 事件监听器cloneNode() 不会复制通过 addEventListener 绑定的事件。

  3. 表单值:部分表单元素(如 <input>)的值可能被克隆,具体行为因浏览器而异。


七、最佳实践与性能优化

  1. 批量操作减少重绘
    使用 DocumentFragment 进行多次操作后一次性插入:

    const fragment = document.createDocumentFragment();
    for (let i = 0; i < 100; i++) {
      const li = document.createElement("li");
      li.textContent = `Item ${i}`;
      fragment.appendChild(li);
    }
    document.getElementById("list").appendChild(fragment);
  2. 谨慎操作 DOM

    • 避免频繁操作 DOM(如循环内直接插入节点)。

    • 必要时先隐藏元素(display: none),操作完成后再显示。


示例:

<div class="box w">
  <div class="box-hd">
      <h3>精品推荐</h3>
      <a href="#">查看全部</a>
  </div>
  <div class="box-bd">
      <ul class="clearfix">

      </ul>
  </div>
</div>
<script>
  // 1. 重构  
  let data = [
      {
          src: 'images/course01.png',
          title: 'Think PHP 5.0 博客系统实战项目演练',
          num: 1125
      },
      {
          src: 'images/course02.png',
          title: 'Android 网络动态图片加载实战',
          num: 357
      },
      {
          src: 'images/course03.png',
          title: 'Angular2 大前端商城实战项目演练',
          num: 22250
      },
      {
          src: 'images/course04.png',
          title: 'Android APP 实战项目演练',
          num: 389
      },
      {
          src: 'images/course05.png',
          title: 'UGUI 源码深度分析案例',
          num: 124
      },
      {
          src: 'images/course06.png',
          title: 'Kami2首页界面切换效果实战演练',
          num: 432
      },
      {
          src: 'images/course07.png',
          title: 'UNITY 从入门到精通实战案例',
          num: 888
      },
      {
          src: 'images/course08.png',
          title: 'Cocos 深度学习你不会错过的实战',
          num: 590
      },
  ]
  const ul = document.querySelector('.box-bd ul')
  // 1. 根据数据的个数,创建 对应的小li
  for (let i = 0; i < data.length; i++) {
      // 2. 创建新的小li
      const li = document.createElement('li')
      // 把内容给li
      li.innerHTML = `
          <a href="#">
              <img src=${data[i].src} alt="">
              <h4>
                  ${data[i].title}
              </h4>
              <div class="info">
                  <span>高级</span> • <span>${data[i].num}</span>人在学习
              </div>
          </a>
      `
      // 3. ul追加小li
      ul.appendChild(li)
  }
</script>

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

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

相关文章

快速求平方根

1. 前置知识 建议首先阅读我的另外一篇文章《雷神之锤 III 竞技场》快速求平方根倒数的计算探究》。建议大家自己看过《雷神之锤 III 竞技场》快速求平方根倒数的计算探究》学会快速求平方根倒数算法后&#xff0c;不看我这篇文章&#xff0c;自己推导一篇快速求平方根的算法&…

C语言基础要素(019):输出ASCII码表

计算机以二进制处理信息&#xff0c;但二进制对人类并不友好。比如说我们规定用二进制值 01000001 表示字母’A’&#xff0c;显然通过键盘输入或屏幕阅读此数据而理解它为字母A&#xff0c;是比较困难的。为了有效的使用信息&#xff0c;先驱者们创建了一种称为ASCII码的交换代…

CI/CD(九) Jenkins共享库与多分支流水线准备

后端构建 零&#xff1a;安装插件 Pipeline: Stage View&#xff08;阶段视图&#xff09;、SSH Pipeline Steps&#xff08;共享库代码中要调用sshCommond命令&#xff09; 一、上传共享库 二、Jenkins配置共享库 3、新增静态资源与修改配置 如果是docker和k8s启动&#xf…

pip安装timm依赖失败

在pycharm终端给虚拟环境安装timm库失败&#xff08; pip install timm&#xff09;&#xff0c;提示你要访问 https://rustup.rs/ 来下载并安装 Rust 和 Cargo 直接不用管&#xff0c;换一条命令 pip install timm0.6.13 成功安装 简单粗暴

详解隔离级别(4种),分别用表格展示问题出现的过程及解决办法

选择隔离级别的时候&#xff0c;既需要考虑数据的一致性&#xff0c;避免脏数据&#xff0c;又要考虑系统性能的问题。下面我们通过商品抢购的场景来讲述这4种隔离级别的区别 未提交读&#xff08;read uncommitted&#xff09; 未提交读是最低的隔离级别&#xff0c;其含义是…

NO.63十六届蓝桥杯备战|基础算法-⼆分答案|木材加工|砍树|跳石头(C++)

⼆分答案可以处理⼤部分「最⼤值最⼩」以及「最⼩值最⼤」的问题。如果「解空间」在从⼩到⼤的「变化」过程中&#xff0c;「判断」答案的结果出现「⼆段性」&#xff0c;此时我们就可以「⼆分」这个「解空间」&#xff0c;通过「判断」&#xff0c;找出最优解。 这个「⼆分答案…

深层储层弹塑性水力裂缝扩展机理

弹性与弹塑性储层条件下裂缝形态对比 参考&#xff1a; The propagation mechanism of elastoplastic hydraulic fracture in deep reservoir | International Journal of Coal Science & Technology

循环神经网络 - 机器学习任务之异步的序列到序列模式

前面我们学习了机器学习任务之同步的序列到序列模式&#xff1a;循环神经网络 - 机器学习任务之同步的序列到序列模式-CSDN博客 本文我们来学习循环神经网络应用中的第三种模式&#xff1a;异步的序列到序列模式&#xff01; 一、基本概述&#xff1a; 异步的序列到序列模式…

什么是检索增强生成(RAG)

1、什么是检索增强生成&#xff08;RAG&#xff09; 1.1 检索增强生成的概念 检索增强生成&#xff08;Retrieval-Augmented Generation, RAG&#xff09;是一种结合了信息检索和文本生成技术的新型自然语言处理方法。这种方法增强了模型的理解和生成能力。 相较于经典生成…

MATLAB 控制系统设计与仿真 - 33

状态反馈控制系统 -全维状态观测器的实现 状态观测器的建立解决了受控系统不能测量的状态重构问题&#xff0c;使得状态反馈的工程实现成为可能。 考虑到系统的状态方程表达式&#xff0c;如果{A,B}可控&#xff0c;{A,C}可观&#xff0c;且安装系统的性能指标&#xff0c;可…

企业管理系统的功能架构设计与实现

一、企业管理系统的核心功能模块 企业管理系统作为现代企业的中枢神经系统&#xff0c;涵盖了多个核心功能模块&#xff0c;以确保企业运营的顺畅与高效。这些功能模块通常包括&#xff1a; 人力资源管理模块&#xff1a;负责员工信息的录入、维护、查询及统计分析&#xff0c…

覆盖学术、职场、生活的专业计算工具

软件介绍 今天要给大家介绍一款超给力的工具软件——CalcKit 计算器。它就像是你口袋里的智能计算专家&#xff0c;轻松化解日常生活中的各类计算难题。无论是简单的数字加减乘除&#xff0c;还是复杂的专业运算&#xff0c;它都不在话下。 这款软件内置了极为强大的计算功能…

【大模型系列篇】大模型基建工程:基于 FastAPI 自动构建 SSE MCP 服务器 —— 进阶篇

&#x1f525;&#x1f525;&#x1f525; 上期 《大模型基建工程&#xff1a;基于 FastAPI 自动构建 SSE MCP 服务器》中我们使用fastapi-mcp自动挂载fastapi到mcp工具&#xff0c;通过源码分析和实践&#xff0c;我们发现每次sse请求又转到了内部fastapi RESTful api接口&…

【python】Plot a Square

文章目录 1、功能描述2、代码实现3、效果展示4、完整代码5、涉及到的库函数 更多有趣的代码示例&#xff0c;可参考【Programming】 1、功能描述 用 python 实现&#xff0c;以 A和B两个点为边长&#xff0c;方向朝 C 绘制正方形 思路&#xff1a; 计算向量 AB 和 AC。使用向…

实战打靶集锦-37-Wpwnvm

文章目录 1. 主机发现2. 端口扫描&服务枚举3. 服务探查4. 系统提权 靶机地址&#xff1a;https://download.vulnhub.com/wpwn/wpwnvm.zip 1. 主机发现 目前只知道目标靶机在192.168.37.xx网段&#xff0c;通过如下的命令&#xff0c;看看这个网段上在线的主机。 $ nmap -…

三、GPIO

一、GPIO简介 GPIO&#xff08;General Purpose Input Output&#xff09;通用输入输出口GPIO引脚电平&#xff1a;0V&#xff08;低电平&#xff09;~3.3V&#xff08;高电平&#xff09;&#xff0c;部分引脚可容忍5V 容忍5V&#xff0c;即部分引脚输入5V的电压&#xff0c;…

混杂模式(Promiscuous Mode)与 Trunk 端口的区别详解

一、混杂模式&#xff08;Promiscuous Mode&#xff09; 1. 定义与工作原理 定义&#xff1a;混杂模式是网络接口的一种工作模式&#xff0c;允许接口接收通过其物理链路的所有数据包&#xff0c;而不仅是目标地址为本机的数据包。工作层级&#xff1a;OSI 数据链路层&#x…

[dp5_多状态dp] 按摩师 | 打家劫舍 II | 删除并获得点数 | 粉刷房子

目录 1.面试题 17.16. 按摩师 题解 2.打家劫舍 II 题解 3.删除并获得点数 题解 4.粉刷房子 题解 一定要有这样的能力&#xff0c;碰到一个新题的时候&#xff0c;可以往之前做过的题方向靠&#xff01; 打家劫舍问题模型: 不能选择相邻的两个数&#xff0c;并且要最终…

DM数据库配置归档模式的两种方式

归档模式&#xff0c;联机日志文件中的内容保存到硬盘中&#xff0c;形成归档日志文件(REDO日志)。 采用归档模式会对系统的性能产生些许影响&#xff0c;然而系统在归档模式下运行会更安全&#xff0c;当 出现故障时其丢失数据的可能性更小&#xff0c;这是因为一旦出现介质故…

Agent TARS与Manus的正面竞争

Agent TARS 是 Manus 的直接竞争对手&#xff0c;两者在 AI Agent 领域形成了显著的技术与生态对抗。 一、技术架构与功能定位的竞争 集成化架构 vs 模块化设计 Agent TARS 基于字节跳动的 UI-TARS 视觉语言模型&#xff0c;将视觉感知、推理、接地&#xff08;grounding&#…