作者:简生

链接:https://www.zhihu.com/question/24702250/answer/28695133

来源:知乎

1. W3C 标准

querySelectorAll 属于 W3C 中的 Selectors API 规范 [1]。而 getElementsBy 系列则属于 W3C 的 DOM 规范 [2]。

2. 浏览器兼容

querySelectorAll 已被 IE 8+、FF 3.5+、Safari 3.1+、Chrome 和 Opera 10+ 良好支持 。

getElementsBy 系列,以最迟添加到规范中的 getElementsByClassName 为例,IE 9+、FF 3 +、Safari 3.1+、Chrome 和 Opera 9+ 都已经支持该方法了。

3. 接收参数

querySelectorAll 方法接收的参数是一个 CSS 选择符。而 getElementsBy 系列接收的参数只能是单一的className、tagName 和 name。代码如下 [3]:

varc1=document.querySelectorAll('.b1 .c');varc2=document.getElementsByClassName('c');varc3=document.getElementsByClassName('b2')[0].getElementsByClassName('c');

需要注意的是,querySelectorAll 所接收的参数是必须严格符合 CSS 选择符规范的。所以下面这种写法,将会抛出异常。代码如下 [4]:

try{vare1=document.getElementsByClassName('1a2b3c');vare2=document.querySelectorAll('.1a2b3c');}catch(e){console.error(e.message);}console.log(e1&&e1[0].className);console.log(e2&&e2[0].className);

(CSS 选择器中的元素名,类和 ID 均不能以数字为开头。)

4. 返回值

大部分人都知道,querySelectorAll 返回的是一个 Static Node List,而 getElementsBy 系列的返回的是一个 Live Node List。

看看下面这个经典的例子 [5]:

// Demo 1varul=document.querySelectorAll('ul')[0],lis=ul.querySelectorAll("li");for(vari=0;i

因为 Demo 2 中的 lis 是一个动态的 Node List, 每一次调用 lis 都会重新对文档进行查询,导致无限循环的问题。

而 Demo 1 中的 lis 是一个静态的 Node List,是一个 li 集合的快照,对文档的任何操作都不会对其产生影响。

但为什么要这样设计呢?

其实,在 W3C 规范中对 querySelectorAll 方法有明确规定 [6]:

The NodeList object returned by the querySelectorAll() methodmust be static([DOM], section 8).

那什么是 NodeList 呢?

W3C 中是这样说明的 [7]:

The NodeList interface provides the abstraction of an ordered collection of nodes, without defining or constraining how this collection is implemented.NodeList objects in the DOM are live.

所以,NodeList 本质上是一个动态的 Node 集合,只是规范中对 querySelectorAll 有明确要求,规定其必须返回一个静态的 NodeList 对象。

我们再看看在 Chrome 上面是个什么样的情况:

document.querySelectorAll('a').toString();// return "[object NodeList]"document.getElementsByTagName('a').toString();// return "[object HTMLCollection]"

这里又多了一个 HTMLCollection 对象出来,那 HTMLCollection 又是什么?

HTMLCollection 在 W3C 的定义如下 [8]:

An HTMLCollection is a list of nodes. An individual node may be accessed by either ordinal index or the node's name or id attributes.

Note: Collections in the HTML DOM are assumed to be live meaning that they are automatically updated when the underlying document is changed.

实际上,HTMLCollection 和 NodeList 十分相似,都是一个动态的元素集合,每次访问都需要重新对文档进行查询。两者的本质上差别在于,HTMLCollection 是属于Document Object Model HTML规范,而 NodeList 属于Document Object Model Core规范。

这样说有点难理解,看看下面的例子会比较好理解 [9]:

varul=document.getElementsByTagName('ul')[0],lis1=ul.childNodes,lis2=ul.children;console.log(lis1.toString(),lis1.length);// "[object NodeList]" 11console.log(lis2.toString(),lis2.length);// "[object HTMLCollection]" 4

NodeList 对象会包含文档中的所有节点,如 Element、Text 和 Comment 等。

HTMLCollection  对象只会包含文档中的 Element 节点。

另外,HTMLCollection 对象比 NodeList 对象 多提供了一个 namedItem 方法。

所以在现代浏览器中,querySelectorAll 的返回值是一个静态的 NodeList 对象,而 getElementsBy 系列的返回值实际上是一个 HTMLCollection 对象 。

最新文章

  1. 【夯实PHP基础系列】linux下yum安装PHP APC
  2. 同 一个页面,不同请求路径,如何根据实际场景写JS
  3. Leetcode 21 Merge Two Sorted Lists 链表
  4. 《Apache负载均衡》RHEL6
  5. APMServ5.2.6 + xdebug + notepad++调试环境配置
  6. 51nod建设国家
  7. Silverlight闹钟
  8. IOS 通过button获取cell
  9. [学习笔记] 多项式与快速傅里叶变换(FFT)基础
  10. select设置disable后ie修改默认字体颜色暂时解决
  11. solr安装血泪史
  12. 一、Python表达式基础
  13. idea 使用debugger技巧
  14. qt系统托盘显示、无主窗体
  15. 没有upcast 也不会发生多态
  16. sboot mybatis
  17. C语言 链表(VS2012版)
  18. Django01-Django基础
  19. [EXP]Microsoft Windows CONTACT - Remote Code Execution
  20. 转《canvas实现滤镜效果》

热门文章

  1. 插入外置网卡端口顺序混乱--linux系统
  2. 基于Vue的WebApp项目开发(六)
  3. 从golang-gin-realworld-example-app项目学写httpapi (六)
  4. Java学习---InetAddress类的学习
  5. ASP.NET Core 返回文件、用户下载文件,从网站下载文件,动态下载文件
  6. scala基础入门
  7. POI读取单元格信息及单元格公式
  8. windows server 2008/2012安装PostgreSQL过程及问题总结
  9. sed.md
  10. swift 第一个IOS应用程序