`
lvwenwen
  • 浏览: 959141 次
  • 性别: Icon_minigender_1
  • 来自: 魔都
社区版块
存档分类
最新评论

javascript:像操作Array一样操作NodeList

阅读更多
javascript:像操作Array一样操作NodeList

  在web前端编程中,我们通常会通过document.getElementsByTagName的方法取出一组相同标签的dom元素,比如:
var anchors = document.getElementsByTagName("a");

for (i = 0; i < anchors.length; i++) {

var ele=anchors[i];//取某一个元素
//some code here

}
上面的代码表示获取文档中的所有链接元素,然后遍历做一些事情。
      也许你会问,通过这种方法获取的这一组dom元素不就是一个数组吗?你看,你都可以直接获取它的length属性,还可以根据索引取到对应的单独元素,根据大牛的著名鸭子理论,它像鸭子一样行走(有length属性),像鸭子一样叫唤(根据索引取值),那么它就是一只鸭子。结论不言自明了吧?
      如果,你已经对javascript稍微有过深入的了解,有length属性,可以索引取值,一定是数组吗,好像arguments也会这么一手吧,arguments是数组?虽然在实际开发的时候,我们把它当做普通数组来操作,length和for循环使用的不亦乐乎,而且并不见得会出错。
      但是,它真的不是数组(Array),而是NodeList。NodeList不是数组。
       What a surprise,right?

1、NodeList为什么不是数组?

验证NodeList是不是数组,最直接的方法也许是试一下Array专有的push和pop大法:
var anchors = document.getElementsByTagName("a");

var newEle = document.createElement("a");//新建一个a元素

anchors.push(newEle);//push

var element= anchors.pop();//pop
您可以自己测试一下,上面的代码不管是push还是pop方法,无一例外的会提示你没有push或者pop方法。还有疑问吗?这样就结束了吗?这种片面的测试反倒使楼猪无法高枕无忧心安理得了。我们完全可以像证明arguments不是数组一样,也用同样的方法证明NodeList不是数组。看下面的代码吧:
Array.prototype.testNodeList = "test nodelist"; //数组添加原型属性

function funcNodeList() {
var links = document.getElementsByTagName("a");
alert(links.testNodeList);
}

function test() {
alert(new Array().testNodeList); //test nodelist
funcNodeList(); //#ff0000? what the hell is that?
}
test(); //测试一下

通过上面的分析,我们可以肯定NodeList不是数组(Array)了。那么如何按照我们操作集合的习惯操作NodeList呢?

2、像操作Array一样操作NodeList

既然NodeList有length,可以for循环索引取值,转换成数组还不是轻而易举?哈哈,最直接的思路是这样的:
var arr = new Array();

var anchors = document.getElementsByTagName("a")

for (var i = 0; i < anchors.length; i++) {

var ele = anchors[i];

arr.push(ele); //arr就是我们要的数组

}
简明扼要说明一下吧:先new一个Array,遍历NodeList,然后将每一个单独的元素push到数组变量里,最后操作数组变量,over。有没有智商受辱的感觉?
上面不是跟您开玩笑,因为下面是楼猪在网上google到的,两行代码就可以将NodeList转换成Array来使用了:
var anchors = document.getElementsByTagName("a");

var arr = Array.prototype.slice.call(anchors); //非ie浏览器正常
【类数组对象如arguments,nodelist等转换为数组都可以采用slice方法。楼猪补注】但是,最最遗憾的事情发生了:上面的代码在万恶的IE下不能正常工作,IE会给你提示: 缺少 JScript 对象。
       你可能会对上面的一大段分析不屑一顾,认为没有必要将NodeList转换成Array来操作。其实,楼猪个人也认为,不管在哪种编程语言里,类型转换都是非常不明智的行为。最常见的比如c#里的装箱和拆箱,数值型数据转换,有性能问题,一不小心还会触雷。但是为什么楼猪单独要把NodeList当做Array来处理呢?因为动态改变NodeList的时候,直接操作NodeList很可能会误闯禁区而浑然不觉。下面举个例子:
(1)、html文档片段
<div id="divAnchor">
        <a href="http://www.cnblogs.com/jeffwongishandsome/">link test</a>
</div>
复制代码
(2)、javascript测试代码
var anchors = document.getElementsByTagName("a");

for (i = 0; i < anchors.length; i++) {

var ele= document.createElement("a");

ele.setAttribute("href", "http://www.cnblogs.com/jeffwongishandsome/");

ele.appendChild(document.createTextNode("new link test"));

document.getElementById("divAnchor").appendChild(ele); //div附加一个新链接

}
在文档加载结束后,执行上面的脚本。我们的本意是在div内,已经存在的a元素后再附加一个a元素。但是,您可以运行一下,浏览器crash掉了吧?楼猪这里IE直接挂掉,FF提示脚本正忙,是否停止脚本运行,点击停止后,页面内已经生成了n多个a链接。其实我们可以大胆分析出原因来:for循环NodeList(前提:for循环内部添加了新的元素使nodelist长度发生了变化。感谢陈童鞋超群的建议),它的length会不断变化上升,循环循环再循环,最后成了个死循环。而用下面的代码,和我们预期的效果是一样一样的:
var links = document.getElementsByTagName("a");

var anchors = null; //数组

try {

anchors = Array.prototype.slice.call(links);

}

catch (e) { //兼容ie

anchors = new Array();

for (var i = 0; i < links.length; i++) {

anchors.push(links[i]);

}

}

for (i = 0; i < anchors.length; i++) { //数组循环 安全多了

var ele = document.createElement("a");

ele.setAttribute("href", "http://www.cnblogs.com/jeffwongishandsome/");

ele.appendChild(document.createTextNode("new link test"));

document.getElementById("divAnchor").appendChild(ele); //div附加一个新链接

}

那么你可能会问,不转换不行吗?没有那么死板,当然是可以的,只要对我们平时熟悉的编码习惯稍微动点小手术就可以了:
var anchors = document.getElementsByTagName("a");

var len = anchors.length; //定义一个变量

for (i = 0; i < len; i++) { //对局部变量len进行循环

var ele = document.createElement("a");

ele.setAttribute("href", "http://www.cnblogs.com/jeffwongishandsome/");

ele.appendChild(document.createTextNode("new link test"));

document.getElementById("divAnchor").appendChild(ele); //div附加一个新链接

}

到这里,不管有无疑问,实际编程如何取舍,楼猪都要感谢您的阅读了。期待指点。

作者:Jeff Wong
出处:http://jeffwongishandsome.cnblogs.com/
本文版权归作者和博客园共有,欢迎围观转载。转载时请您务必在文章明显位置给出原文链接,谢谢您的合作。
分类: javascript
分享到:
评论

相关推荐

    在javascript将NodeList作为Array数组处理的方法

    在JavaScript中,我们经常需要操作DOM元素集合。document.getElementsByTagName方法是获取具有相同标签名的DOM元素列表的一种常用方式。这个方法返回的是一个NodeList对象,它与数组有相似之处,例如都有length属性...

    dojo命令和nodeList的详细介绍

    要对NodeList进行操作,你需要遍历其项,或者使用Array.prototype上的方法,如`Array.from(nodeList)`将其转换为真正的数组。NodeList是“活”的,意味着当文档中的DOM结构发生变化时,NodeList会自动更新。 Dojo与...

    08-HTMLCollection-NodeList-区别.md

    HTMLCollection和NodeList是前端开发中经常接触到的类数组对象,它们在DOM操作中扮演着重要的角色,对于开发者理解DOM结构和编写高效代码至关重要。接下来将从多个角度分析它们的定义、用途以及区别。 ### ...

    js中将HTMLCollection NodeList 伪数组转换成数组的代码.docx

    在JavaScript开发过程中,经常需要处理DOM操作,这通常涉及到对`HTMLCollection`、`NodeList`以及所谓的“伪数组”进行操作。这些对象虽然表现得像数组,但并不具备数组的所有特性,比如它们没有`push`、`pop`等数组...

    前端面试题之arrayLike.zip

    ArrayLike对象不是JavaScript的内置类型,而是一个通用术语,用来描述那些具有类似于数组的行为但并非Array实例的结构。例如,arguments对象和DOM方法返回的NodeList都是ArrayLike的例子。这些对象拥有length属性,...

    使用JS实现JQ查找节点功能的一个demo

    - JavaScript:`document.getElementsByClassName("myClass")`,返回的是NodeList,而不是数组,可以转换为数组使用`Array.from()`。 3. **根据标签名查找**: - jQuery:`$("div")` - JavaScript:`document....

    原生JS实现循环Nodelist Dom列表的4种方式示例

    在JavaScript中,Nodelist对象是DOM操作中常见的一种类型,它表示一组节点,通常是由`querySelectorAll`等方法返回的结果。然而,Nodelist并不像数组那样具有内置的迭代方法,因此在需要遍历这些节点时,我们需要...

    javascript完全学习手册1 源码

    javascript完全自学手册 目 录 第1篇 JavaScript基础篇 第1章 JavaScript简介 1 1.1 JavaScript概述 1 1.1.1 什么是JavaScript 1 1.1.2 JavaScript的基本特点 2 1.1.3 常用的Web开发语言 3 1.2 JavaScript的应用 4 ...

    js中将HTMLCollection/NodeList/伪数组转换成数组的代码

    在JavaScript中,我们经常会遇到需要将HTMLCollection、NodeList或者伪数组(也就是通常所说的类数组对象)转换成真正的数组的情况。类数组对象并不是真正的数组类型,它们通常是函数的参数对象(比如arguments对象...

    ECMAScript 6即将带给我们新的数组操作方法前瞻

    类数组对象是指拥有`length`属性和索引属性的对象,如DOM操作返回的NodeList。可迭代对象则是可以通过迭代器遍历的对象,如Map和Set。 示例: ```javascript let lis = document.querySelectorAll('ul.fancy li'...

    用标准javascript 替代jQuery.

    jQuery 的 `$` 函数返回的对象支持链式操作,而原生 JavaScript 中可以使用 `Array.prototype.forEach` 或 `for...of` 循环来遍历元素集合。 6. **动画效果** jQuery 的动画功能可以通过 CSS3 动画或者使用 `...

    javascript Array数组对象的扩展函数代码

    标题提到的“javascript Array数组对象的扩展函数代码”就是关于如何给Array对象添加额外的方法,以实现更便捷的操作。描述中提到了“去除字符串空格”和“数组排序”,这些都是常见的数组操作。 首先,我们来讨论...

    dom javascript

    4. **DOM遍历与查询**:掌握更高效的查询方法,如`querySelector`、`querySelectorAll`,以及`NodeList`和`Array`的转换技巧,以优化代码性能。 5. **DOM操作与更新**:了解如何改变元素的属性、样式,以及如何插入...

    JavaScript数组.pdf

    JavaScript中存在类数组对象的概念,比如函数的arguments对象或者DOM操作返回的NodeList对象,它们拥有length属性和数字索引,但不是数组实例。可以使用Array.prototype.slice.call()方法将类数组对象转换为数组。 ...

    javascript_使用DOM笔记1

    根据提供的文件信息,我们可以总结出以下几个关键的知识点: ...以上就是关于如何使用DOM进行元素获取和操作的一些基础知识。这些方法在实际开发中非常常见且实用,掌握它们将有助于更好地理解和使用JavaScript与DOM。

    JAVASCRIPT学习总结1.pdf

    4. **数组对象(Array)**:JavaScript的数组是动态大小的数据集合,可以存储任意类型的值。它们提供了丰富的内置方法,如`push()`, `pop()`, `slice()`, `forEach()`等,方便我们操作和处理数据。 5. **字符串对象...

    Looping through a HTML list with JavaScript.zip

    JavaScript提供了多种方法来访问和操作这些元素。 1. **使用DOM遍历** - `getElementsByTagName`: 这是获取页面上所有特定类型的元素的方法。例如,`document.getElementsByTagName('li')`将返回一个`NodeList`...

    Javascript-DOM编程艺术研究.docx

    - **内建对象(Native Objects)**,如Array、Math、Date等,是JavaScript语言内置的对象,可以直接使用。 - **宿主对象(Host Objects)**,如`window`对象,由浏览器提供,提供了与浏览器交互的能力,如`window....

    JavaScript数据结构案例.pdf

    4. 数组的变种:如关联数组(Array-like Objects),如`arguments`对象,以及类数组对象,如DOM方法返回的NodeList。 5. Map和Set:ES6引入的新数据结构,Map用于存储键值对,键可以是任何类型,而Set则用于存储不...

    javascript面试题汇总

    JavaScript面试题汇总涵盖了广泛的知识点,以下是其中一些关键点的详细说明: 1. **变量声明**:在JavaScript中,可以使用var关键字声明变量。在给定的题目中,选项A `( )` 是一个无效的声明,因为没有提供任何值,...

Global site tag (gtag.js) - Google Analytics