你可能没有听说过 js中的 DOM操作还有这个: HTMLCollection 和 NodeList

简介: 该文章详细解释了JavaScript中HTMLCollection和NodeList这两种DOM集合类型的特性、使用方法及其区别,并通过实例代码展示了如何操作这两种集合来选取和遍历DOM元素。

HTMLCollection和NodeList

⛱️序言

犹记得上回有一次面试的时候被问到 htmlCollection 和数组的关系。当时的我心里想的是, html 中的集合和数组的关系,那解题就是说 set 和数组的关系?

于是……面试官当下立即纠正我, htmlCollection 是单词连在一起的,是 js 的一个内容。当时我可能是想……找个地洞钻……

懵逼

不会的咋还是得虚心接收,查漏补缺还是比较合理。经过一系列的资料查询和总结之后,得出以下总结。

下面开始进入本文的讲解~🙋

🎈一、基础知识

1. 定义

(1)HTMLCollection

  • HTMLCollection 表示一个包含了元素(元素顺序为文档流中的接口)的集合(通用集合),还提供了从该集合中选择元素的属性和方法。
  • 例如使用 getElementsByTagName() 方法返回的就是一个 HTMLCollection 对象。

(2)NodeList

  • NodeList 对象是节点的集合。

  • 通过以下方法,可以获取到 NodeList 对象。主要有:在一些旧版本浏览器中的方法,比如 getElementsClassName() 方法,返回的是 NodeList 对象,而不是 HTMLCollection 对象。所有浏览器的 Node.childNodes 属性返回的都是 NodeList 对象。大部分浏览器的 document.querySelectorAll() 返回的是 NodeList 对象。

  • 我们可以通过以下代码进行验证:

    document.body.childNodes instanceof NodeList // true
    document.querySelector('body') instanceof NodeList // true or false
    document.getElementByClassName('body') instanceof NodeList // true or false
    

2. 属性和方法

(1)HTMLCollection

HTMLCollection 对象中的属性和方法,举例两个:

  • item(index) —— 返回 HTMLCollection指定索引的元素,如果不存在则返回 null
  • length (只读) —— 返回 HTMLCollection 中元素的数量

接下来用一个例子来进行阐述验证。具体代码如下:

document.getElementsByTagName('body') instanceof HTMLCollection // true

const htmlCollection = document.getElementsByTagName('body')
console.log(htmlCollection.item(0)) // <body>...</body>
console.log(htmlCollection.length()) // 1

(2)NodeList

NodeList 对象中的属性和方法主要有以下几种。包括:

  • item() —— 返回某个元素基于文档数的索引;
  • length() —— 返回 NodeList 中的节点数量;
  • NodeList.forEach() —— 该方法用于遍历 NodeList 中的所有成员。它接收一个回调函数作为参数,每遍历一回就要执行这个回调函数一次,用法与数组实例的 forEach 方法是完全一致的;
  • NodeList.keys()/values()/entries() —— 对于这三个方法来说,它们都会返回一个 ES6遍历器对象,可以通过 for…of… 来进行循环遍历,以便于获取每一个成员的信息

同时,需要注意以上三者的区别,分别如下:

  • keys() —— 指的是返回键名的遍历器;
  • values() —— 指的是返回键值的遍历器;
  • entries() —— 指的是返回的遍历器要同时包含键名和键值的信息。

我们写点代码来论证以上内容。具体代码如下:

const nodelist = document.querySelectorAll('body')

console.log(nodelist.item(0)) // <body>...</body>
console.log(nodelist.length) // 1
console.log(nodelist.forEach(item => console.log(item))) // <body>...</body>

for(var key of nodelist.keys()) {
   
  console.log(nodelist[key]) // <body>...</body>
}

for(var value of nodelist.values()) {
   
  console.log(value) // <body>...</body>
}

for(var entry of nodelist.entries()) {
   
  console.log(entry) // [0, body]
}

🪁二、异同点

1. HTMLCollection 与 NodeList 的区别

HTMLCollection NodeList
集合 元素的集合 节点的集合
静态和动态 HTMLCollection 是动态绑定的,是一个动态集合。DOM 树发生变化,HTMLCollection 也会随之变化,说明其节点的增删是敏感的 NodeList 是一个静态集合,其不受 DOM 树元素变化的影响;相当于是 DOM 树、节点数量和类型的快照,也就是说对节点进行增删操作时,NodeList 是感觉不到的。但是对节点内部内容修改,是可以感觉得到的,比如修改 innerHTML
节点 不包含属性节点和文本节点 只有 NodeList 对象有包含属性节点文本节点
元素获取方式 HTMLCollection 元素可以通过 name,id 或 index 索引来获取 NodeList 只能通过 index 索引来获取
伪数组 HTMLCollection 和 NodeList 都是类数组,不是数组,只是长得像数组而已。所以无法使用数组的方法,比如: pop(),push(),或 join() 等等 与 HTMLCollection 一样

2. querySelectorAll 和 getElementsByTagName 的区别

在上述的表格中我们可以了解到, HTMLCollection 是动态集合,当 DOM 树发生变化时, HTMLCollection 也会随之改变。而 NodeList 是静态集合,当 DOM 树发生变化时, NodeList 不会受到 DOM 树变化的影响。我们来举两个例子进行阐述说明。

(1)querySelectorAll

先附上代码:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>htmlCollection和NodeList</title>
</head>
<body>
    <ul>
        <li></li>
        <li></li>
        <li>MondayLab</li>
        <li></li>
        <li></li>
    </ul>
    <script>
        var oldUl = document.querySelectorAll('ul')[0];
        var oldLi = document.querySelectorAll('li');

        console.log(oldUl); // NodeList[body]
        console.log(oldLi.length); // 5

        var newLi = document.createElement('li');
        oldUl.appendChild(newLi);

        console.log(oldLi.length); // 5
        var length = oldLi.length - 1;
        oldLi[length].innerHTML = 'monday';
        console.log(oldLi[length].innerHTML); // monday 
        console.log(oldLi.length); // 5 → 虽然通过innerHTML可以读到文本节点的内容,但是长度依旧不变,还是5
    </script>
</body>
</html>

大家可以看到,使用 querySelectorAll 来获取 <li> ,返回的是一个 NodeList 的集合。且当页面 DOM 结构发生改变时,其长度不会发生任何的改变。但是当改变 innerHTML 时,则文本节点的内容会发生改变。值得注意的是,即使 innerHTML 改变 DOM 的结构,但长度依旧是不变的,还是 5

(2)getElementsByTagName

先附上代码:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>HTMLCollection</title>
</head>
<body>
    <ul>
        <li></li>
        <li></li>
        <li>MondayLab</li>
        <li></li>
        <li></li>
    </ul>
    <script>
        var oldUl = document.getElementsByTagName('ul')[0];
        var oldLi = document.getElementsByTagName('li');

        console.log(oldUl); // HTMLCollection[body]
        console.log(oldLi.length); // 5

        var newLi = document.createElement('li');
        oldUl.appendChild(newLi);

        console.log(oldLi.length); // 6 → 顺利改变 DOM 的结构
    </script>
</body>
</html>

大家可以看到,使用 getElementByTagName 时,获取的是一个 HTMLCollection 集合,这个时候说明对节点进行增删操作时,页面的 DOM 结构会发生改变,且HTMLCollection实时地获取到集合的长度。

📞三、结束语

写到这里的时候,不得不感概 jsDOM 操作是如此的强大,也突然就明白了自己之前写代码的时候为啥会遇到那么多坑。如果在学习之初就追溯于原理,可能跳的坑应该就能少很多了。

讲到这里,关于 HTMLCollectionNodeList 的内容就结束啦!希望对大家有帮助~

彩蛋有几篇我看过还比较好理解的文章,有需要可以当扩展知识进行扩充哦~

🐣彩蛋 One More Thing

(:参考资料

lio_zero👉HTMLCollection和NodeList的区别

Snandy👉将HTMLCollection/NodeList/伪数组转换成数组

binyellow👉DOM操作,HTMLCollection、NodeList

爱民👉NodeList 和 HTMLCollection

(:番外篇

  • 关注公众号星期一研究室,第一时间关注优质文章,更多精选专栏待你解锁~

  • 如果这篇文章对你有用,记得留个脚印jio再走哦~

  • 以上就是本文的全部内容!我们下期见!👋👋👋

相关文章
|
10天前
|
弹性计算 人工智能 架构师
阿里云携手Altair共拓云上工业仿真新机遇
2024年9月12日,「2024 Altair 技术大会杭州站」成功召开,阿里云弹性计算产品运营与生态负责人何川,与Altair中国技术总监赵阳在会上联合发布了最新的“云上CAE一体机”。
阿里云携手Altair共拓云上工业仿真新机遇
|
7天前
|
机器学习/深度学习 算法 大数据
【BetterBench博士】2024 “华为杯”第二十一届中国研究生数学建模竞赛 选题分析
2024“华为杯”数学建模竞赛,对ABCDEF每个题进行详细的分析,涵盖风电场功率优化、WLAN网络吞吐量、磁性元件损耗建模、地理环境问题、高速公路应急车道启用和X射线脉冲星建模等多领域问题,解析了问题类型、专业和技能的需要。
2513 16
【BetterBench博士】2024 “华为杯”第二十一届中国研究生数学建模竞赛 选题分析
|
6天前
|
机器学习/深度学习 算法 数据可视化
【BetterBench博士】2024年中国研究生数学建模竞赛 C题:数据驱动下磁性元件的磁芯损耗建模 问题分析、数学模型、python 代码
2024年中国研究生数学建模竞赛C题聚焦磁性元件磁芯损耗建模。题目背景介绍了电能变换技术的发展与应用,强调磁性元件在功率变换器中的重要性。磁芯损耗受多种因素影响,现有模型难以精确预测。题目要求通过数据分析建立高精度磁芯损耗模型。具体任务包括励磁波形分类、修正斯坦麦茨方程、分析影响因素、构建预测模型及优化设计条件。涉及数据预处理、特征提取、机器学习及优化算法等技术。适合电气、材料、计算机等多个专业学生参与。
1520 14
【BetterBench博士】2024年中国研究生数学建模竞赛 C题:数据驱动下磁性元件的磁芯损耗建模 问题分析、数学模型、python 代码
|
2天前
|
存储 关系型数据库 分布式数据库
GraphRAG:基于PolarDB+通义千问+LangChain的知识图谱+大模型最佳实践
本文介绍了如何使用PolarDB、通义千问和LangChain搭建GraphRAG系统,结合知识图谱和向量检索提升问答质量。通过实例展示了单独使用向量检索和图检索的局限性,并通过图+向量联合搜索增强了问答准确性。PolarDB支持AGE图引擎和pgvector插件,实现图数据和向量数据的统一存储与检索,提升了RAG系统的性能和效果。
|
9天前
|
编解码 JSON 自然语言处理
通义千问重磅开源Qwen2.5,性能超越Llama
击败Meta,阿里Qwen2.5再登全球开源大模型王座
545 14
|
1月前
|
运维 Cloud Native Devops
一线实战:运维人少,我们从 0 到 1 实践 DevOps 和云原生
上海经证科技有限公司为有效推进软件项目管理和开发工作,选择了阿里云云效作为 DevOps 解决方案。通过云效,实现了从 0 开始,到现在近百个微服务、数百条流水线与应用交付的全面覆盖,有效支撑了敏捷开发流程。
19282 30
|
9天前
|
人工智能 自动驾驶 机器人
吴泳铭:AI最大的想象力不在手机屏幕,而是改变物理世界
过去22个月,AI发展速度超过任何历史时期,但我们依然还处于AGI变革的早期。生成式AI最大的想象力,绝不是在手机屏幕上做一两个新的超级app,而是接管数字世界,改变物理世界。
464 48
吴泳铭:AI最大的想象力不在手机屏幕,而是改变物理世界
|
1月前
|
人工智能 自然语言处理 搜索推荐
阿里云Elasticsearch AI搜索实践
本文介绍了阿里云 Elasticsearch 在AI 搜索方面的技术实践与探索。
18837 20
|
1月前
|
Rust Apache 对象存储
Apache Paimon V0.9最新进展
Apache Paimon V0.9 版本即将发布,此版本带来了多项新特性并解决了关键挑战。Paimon自2022年从Flink社区诞生以来迅速成长,已成为Apache顶级项目,并广泛应用于阿里集团内外的多家企业。
17526 13
Apache Paimon V0.9最新进展
|
1天前
|
云安全 存储 运维
叮咚!您有一份六大必做安全操作清单,请查收
云安全态势管理(CSPM)开启免费试用
359 4
叮咚!您有一份六大必做安全操作清单,请查收