JavaScript DOM 节点操作
目录
一、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.注意事项
-
动态集合与静态集合:
-
childNodes和children是动态集合(HTMLCollection/NodeList),会随 DOM 变化自动更新。 -
querySelectorAll返回静态NodeList,不会自动更新。
-
-
空值处理:
-
若节点不存在(如
parentNode为null),使用可选链操作符避免报错:const parentId = element.parentNode?.id; // 安全访问
-
-
节点类型判断:
-
非元素节点(如文本、注释)无法使用
children或firstElementChild等方法。
-
-
兼容性:
-
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。
-
若需彻底释放内存,将引用设为
null:element = 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); // 页面添加克隆的列表
克隆注意事项
-
ID 重复:克隆的元素的
id属性与原元素相同,需手动修改以避免冲突。 -
事件监听器:
cloneNode()不会复制通过addEventListener绑定的事件。 -
表单值:部分表单元素(如
<input>)的值可能被克隆,具体行为因浏览器而异。
七、最佳实践与性能优化
-
批量操作减少重绘
使用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); -
谨慎操作 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. 根据数据的个数,创建 对应的小lifor (let i = 0; i < data.length; i++) {// 2. 创建新的小liconst li = document.createElement('li')// 把内容给lili.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追加小liul.appendChild(li)}
</script>
相关文章:
JavaScript DOM 节点操作
目录 一、DOM 节点 节点类型(Node Types) 二、查找节点 1.查找父节点 1. parentNode 2. parentElement 2.查找子节点 1. childNodes 2. children 3. firstChild / lastChild 4. firstElementChild / lastElementChild 3.查找兄弟节点 1. pre…...
快速求平方根
1. 前置知识 建议首先阅读我的另外一篇文章《雷神之锤 III 竞技场》快速求平方根倒数的计算探究》。建议大家自己看过《雷神之锤 III 竞技场》快速求平方根倒数的计算探究》学会快速求平方根倒数算法后,不看我这篇文章,自己推导一篇快速求平方根的算法&…...
科普:One-Class SVM和SVDD
SVM(支持向量机)算法是用于解决二分类问题的,它在样本空间(高维空间)中找一个最优超平面,使得两类数据点中离超平面最近的点(称为支持向量)到超平面的距离最大。 对于极少数“坏样本…...
Vue 3 中按照某个字段将数组分成多个数组
方法一:使用 reduce 方法 const originalArray [{ id: 1, category: A, name: Item 1 },{ id: 2, category: B, name: Item 2 },{ id: 3, category: A, name: Item 3 },{ id: 4, category: C, name: Item 4 },{ id: 5, category: B, name: Item 5 }, ];const grou…...
冒泡排序笔记
核心思想 通过相邻元素的比较和交换,使较大的元素逐渐“浮”到数组的末尾(像气泡从水底冒到水面一样) 基础冒泡排序 public class BubbleSort{public static void bubbleSort(int[] arr){for(int i 0; i < arr.length - 1; i){//冒泡…...
【ABAP】REST/HTTP技术(一)
1、概念 1.1、SAP 如何提供 Http Service 如果要将 SAP 应用程序服务器 (application server)作为 http 服务提供者,需要定义一个类,这个类必须实现 IF_HTTP_EXTENSION 接口。IF_HTTP_EXTENSION 接口只有一个方法 HANDLE_REQUEST。…...
Flutter PopupMenuButton 深度解析:从入门到架构级实战
在移动应用交互设计中,上下文菜单如同隐形的魔法师,在有限屏幕空间中优雅地扩展操作维度。作为Flutter框架中的核心交互组件,PopupMenuButton绝非简单的菜单触发器,其背后蕴含着Material Design的交互哲学、声明式UI的架构智慧以及…...
C语言基础要素(019):输出ASCII码表
计算机以二进制处理信息,但二进制对人类并不友好。比如说我们规定用二进制值 01000001 表示字母’A’,显然通过键盘输入或屏幕阅读此数据而理解它为字母A,是比较困难的。为了有效的使用信息,先驱者们创建了一种称为ASCII码的交换代…...
VSCode开发者工具快捷键
自动生成浏览器文件.html的快捷方式 在文本里输入: ! enter VSCode常用快捷键列表 代码格式化:Shift Alt F向上或向下移动一行:Alt Up 或者 Alt Down快速复制一行代码:Shift Alt Up 或者 Shift Alt Down快速保…...
CI/CD(九) Jenkins共享库与多分支流水线准备
后端构建 零:安装插件 Pipeline: Stage View(阶段视图)、SSH Pipeline Steps(共享库代码中要调用sshCommond命令) 一、上传共享库 二、Jenkins配置共享库 3、新增静态资源与修改配置 如果是docker和k8s启动…...
使用Deployment运行无状态应用
使用Deployment运行无状态应用 文章目录 使用Deployment运行无状态应用[toc]一、工作负载资源与控制器二、ReplicationController、ReplicaSet和Deployment1. ReplicationController(已淘汰)2. ReplicaSet(ReplicationController 的增强版&am…...
pip安装timm依赖失败
在pycharm终端给虚拟环境安装timm库失败( pip install timm),提示你要访问 https://rustup.rs/ 来下载并安装 Rust 和 Cargo 直接不用管,换一条命令 pip install timm0.6.13 成功安装 简单粗暴...
详解隔离级别(4种),分别用表格展示问题出现的过程及解决办法
选择隔离级别的时候,既需要考虑数据的一致性,避免脏数据,又要考虑系统性能的问题。下面我们通过商品抢购的场景来讲述这4种隔离级别的区别 未提交读(read uncommitted) 未提交读是最低的隔离级别,其含义是…...
NO.63十六届蓝桥杯备战|基础算法-⼆分答案|木材加工|砍树|跳石头(C++)
⼆分答案可以处理⼤部分「最⼤值最⼩」以及「最⼩值最⼤」的问题。如果「解空间」在从⼩到⼤的「变化」过程中,「判断」答案的结果出现「⼆段性」,此时我们就可以「⼆分」这个「解空间」,通过「判断」,找出最优解。 这个「⼆分答案…...
深层储层弹塑性水力裂缝扩展机理
弹性与弹塑性储层条件下裂缝形态对比 参考: The propagation mechanism of elastoplastic hydraulic fracture in deep reservoir | International Journal of Coal Science & Technology...
循环神经网络 - 机器学习任务之异步的序列到序列模式
前面我们学习了机器学习任务之同步的序列到序列模式:循环神经网络 - 机器学习任务之同步的序列到序列模式-CSDN博客 本文我们来学习循环神经网络应用中的第三种模式:异步的序列到序列模式! 一、基本概述: 异步的序列到序列模式…...
什么是检索增强生成(RAG)
1、什么是检索增强生成(RAG) 1.1 检索增强生成的概念 检索增强生成(Retrieval-Augmented Generation, RAG)是一种结合了信息检索和文本生成技术的新型自然语言处理方法。这种方法增强了模型的理解和生成能力。 相较于经典生成…...
MATLAB 控制系统设计与仿真 - 33
状态反馈控制系统 -全维状态观测器的实现 状态观测器的建立解决了受控系统不能测量的状态重构问题,使得状态反馈的工程实现成为可能。 考虑到系统的状态方程表达式,如果{A,B}可控,{A,C}可观,且安装系统的性能指标,可…...
PM2 在 Node.js 项目中的使用与部署指南
一、PM2 简介 PM2 是一个带有负载均衡功能的 Node.js 应用程序的进程管理器。它可以让你的 Node.js 应用程序始终保持运行状态,即使出现错误或服务器重启也能自动恢复。同时,它还提供了诸如日志管理、性能监控等实用功能,极大地简化了 Nod…...
企业管理系统的功能架构设计与实现
一、企业管理系统的核心功能模块 企业管理系统作为现代企业的中枢神经系统,涵盖了多个核心功能模块,以确保企业运营的顺畅与高效。这些功能模块通常包括: 人力资源管理模块:负责员工信息的录入、维护、查询及统计分析,…...
RTOS基础 -- NXP M4小核的RPMsg-lite与端点机制回顾
一、RPMsg-lite与端点机制回顾 在RPMsg协议框架中: Endpoint(端点) 是一个逻辑通信端口,由本地地址(local addr)、远程地址(remote addr)和回调函数组成。每个消息都会发送到特定的…...
Cursor的主要好处
以下是Cursor的主要好处: 代码生成与优化 • 快速生成代码:根据简短描述或部分代码片段,Cursor能快速生成完整代码模块,还能智能预测下一步操作,将光标放在合适位置,让开发者一路Tab键顺滑编写代码。 • …...
覆盖学术、职场、生活的专业计算工具
软件介绍 今天要给大家介绍一款超给力的工具软件——CalcKit 计算器。它就像是你口袋里的智能计算专家,轻松化解日常生活中的各类计算难题。无论是简单的数字加减乘除,还是复杂的专业运算,它都不在话下。 这款软件内置了极为强大的计算功能…...
【大模型系列篇】大模型基建工程:基于 FastAPI 自动构建 SSE MCP 服务器 —— 进阶篇
🔥🔥🔥 上期 《大模型基建工程:基于 FastAPI 自动构建 SSE MCP 服务器》中我们使用fastapi-mcp自动挂载fastapi到mcp工具,通过源码分析和实践,我们发现每次sse请求又转到了内部fastapi RESTful api接口&…...
嵌入式硬件篇---USBUART串口
文章目录 前言一、UART 通信原理1.发送原理2.接收原理二、单片机UART接收十六进制数的处理方式1.数据解析2.数据存储3.执行相应操作三、USB通信原理四、USB 转串口通信1.硬件连接2.驱动程序3.数据传输过程五、通信特点与应用场景1.USB通信特点与应用场景2.串口通过特点与应用场…...
湖南移动广东电信DNS
湖南移动DNS: DNS 1: 111.8.14.18 DNS 2: 211.142.211.124 DNS 3: 2409:8050:2000::1 DNS 4: 2409:8050:2000:1000::1 湖南电信DNS: DNS 1: 59.51.78.210 DNS 2: 222.246.129.80 DNS 3: 240e:50:c800::210 DNS 4: 240e:50:5000::80 广东电信DNS: DNS 1…...
百度查询的ip与命令行输入 ipconfig 显示的IP地址有以下主要区别:
IP类型不同 百度中的IP是公网IP(WAN IP),由运营商分配,用于在互联网上标识你的网络出口。 ipconfig 显示的是本地IP(LAN IP),通常是路由器分配给设备的私有地址(如192.168.x.x、10.…...
【python】Plot a Square
文章目录 1、功能描述2、代码实现3、效果展示4、完整代码5、涉及到的库函数 更多有趣的代码示例,可参考【Programming】 1、功能描述 用 python 实现,以 A和B两个点为边长,方向朝 C 绘制正方形 思路: 计算向量 AB 和 AC。使用向…...
实战打靶集锦-37-Wpwnvm
文章目录 1. 主机发现2. 端口扫描&服务枚举3. 服务探查4. 系统提权 靶机地址:https://download.vulnhub.com/wpwn/wpwnvm.zip 1. 主机发现 目前只知道目标靶机在192.168.37.xx网段,通过如下的命令,看看这个网段上在线的主机。 $ nmap -…...
三、GPIO
一、GPIO简介 GPIO(General Purpose Input Output)通用输入输出口GPIO引脚电平:0V(低电平)~3.3V(高电平),部分引脚可容忍5V 容忍5V,即部分引脚输入5V的电压,…...
