文本整理了 javascript 操作 DOM 的一些常用的 api, 根据其作用整理成为创建, 修改, 查询等多种类型的 api, 主要用于复习基础知识, 加深对原生 js 的认识
基本概念
在讲解操作 DOM 的 api 之前, 首先我们来复习一下一些基本概念, 这些概念是掌握 api 的关键, 必须理解它们
Node 类型
DOM1 级定义了一个 Node 接口, 该接口由 DOM 中所有节点类型实现这个 Node 接口在 JS 中是作为 Node 类型实现的在 IE9 以下版本无法访问到这个类型, JS 中所有节点都继承自 Node 类型, 都共享着相同的基本属性和方法
Node 有一个属性 nodeType 表示 Node 的类型, 它是一个整数, 其数值分别表示相应的 Node 类型, 具体如下:
- Node.ELEMENT_NODE:1
- Node.ATTRIBUTE_NODE:2
- Node.TEXT_NODE:3
- Node.CDATA_SECTION_NODE:4
- Node.ENTITY_REFERENCE_NODE:5
- Node.ENTITY_NODE:6
- Node.PROCESSING_INSTRUCTION_NODE:7
- Node.COMMENT_NODE:8
- Node.DOCUMENT_NODE:9
- Node.DOCUMENT_TYPE_NODE:10
- Node.DOCUMENT_FRAGMENT_NODE:11
- Node.NOTATION_NODE:12
假设我们要判断一个 Node 是不是元素, 我们可以这样判断
- if (someNode.nodeType == 1) {
- console.log("Node is a element");
- }
这些 Node 类型中, 我们最常用的就是 element,text,attribute,comment,document,document_fragment 这几种类型
我们简单来介绍一下这几种类型:
Element 类型
Element 提供了对元素标签名, 子节点和特性的访问, 我们常用 html 元素比如 div,span,a 等标签就是 element 中的一种 Element 有下面几条特性:
(1)nodeType 为 1
(2)nodeName 为元素标签名, tagName 也是返回标签名
(3)nodeValue 为 null
(4)parentNode 可能是 Document 或 Element
(5) 子节点可能是 Element,Text,Comment,Processing_Instruction,CDATASection 或 EntityReference
Text 类型
Text 表示文本节点, 它包含的是纯文本内容, 不能包含 html 代码, 但可以包含转义后的 html 代码 Text 有下面的特性:
(1)nodeType 为 3
(2)nodeName 为 #text
(3)nodeValue 为文本内容
(4)parentNode 是一个 Element
(5) 没有子节点
Attr 类型
Attr 类型表示元素的特性, 相当于元素的 attributes 属性中的节点, 它有下面的特性:
(1)nodeType 值为 2
(2)nodeName 是特性的名称
(3)nodeValue 是特性的值
(4)parentNode 为 null
Comment 类型
Comment 表示 HTML 文档中的注释, 它有下面的几种特征:
(1)nodeType 为 8
(2)nodeName 为 #comment
(3)nodeValue 为注释的内容
(4)parentNode 可能是 Document 或 Element
(5) 没有子节点
Document
Document 表示文档, 在浏览器中, document 对象是 HTMLDocument 的一个实例, 表示整个页面, 它同时也是 window 对象的一个属性 Document 有下面的特性:
(1)nodeType 为 9
(2)nodeName 为 #document
(3)nodeValue 为 null
(4)parentNode 为 null
(5) 子节点可能是一个 DocumentType 或 Element
DocumentFragment 类型
DocumentFragment 是所有节点中唯一一个没有对应标记的类型, 它表示一种轻量级的文档, 可能当作一个临时的仓库用来保存可能会添加到文档中的节点 DocumentFragment 有下面的特性:
(1)nodeType 为 11
(2)nodeName 为 #document-fragment
(3)nodeValue 为 null
(4)parentNode 为 null
我们简单地介绍了几种常见的 Node 类型, 要记住, HTML 中的节点并不只是包括元素节点, 它还包括文本节点, 注释节点等等在这里我们只是简单地说明了几种常见的节点, 想要进一步学习的同学可以查找一下相关资料
节点创建型 api
在这里, 我将常用的 DOM 操作 api 进行分类, 首先要介绍的是创建型的 api 这一类型的 api, 简而言之就是用来创建节点的
createElement
createElement 通过传入指定的一个标签名来创建一个元素, 如果传入的标签名是一个未知的, 则会创建一个自定义的标签, 注意: IE8 以下浏览器不支持自定义标签
使用如下:
var div = document.createElement("div");
使用 createElement 要注意: 通过 createElement 创建的元素并不属于 html 文档, 它只是创建出来, 并未添加到 html 文档中, 要调用 appendChild 或 insertBefore 等方法将其添加到 HTML 文档树中
createTextNode
createTextNode 用来创建一个文本节点, 用法如下:
var textNode = document.createTextNode("一个 TextNode");
createTextNode 接收一个参数, 这个参数就是文本节点中的文本, 和 createElement 一样, 创建后的文本节点也只是独立的一个节点, 同样需要 appendChild 将其添加到 HTML 文档树中
cloneNode
cloneNode 是用来返回调用方法的节点的一个副本, 它接收一个 bool 参数, 用来表示是否复制子元素, 使用如下:
- var parent = document.getElementById("parentElement");
- var parent2 = parent.cloneNode(true); // 传入 true
- parent2.id = "parent2";
这段代码通过 cloneNode 复制了一份 parent 元素, 其中 cloneNode 的参数为 true, 表示 parent 的子节点也被复制, 如果传入 false, 则表示只复制了 parent 节点
我们看看这个例子
<div id="parent">
我是父元素的文本
- <br/>
- <span>
我是子元素
- </span>
- </div>
- <button id="btnCopy">
复制
- </button>
- var parent = document.getElementById("parent"); document.getElementById("btnCopy").onclick
- = function(){ var parent2 = parent.cloneNode(true); parent2.id = "parent2";
- document.body.appendChild(parent2); }
这段代码很简单, 主要是绑定 button 事件, 事件内容是复制了一个 parent, 修改其 id, 然后添加到文档中
这里有几点要注意:
(1) 和 createElement 一样, cloneNode 创建的节点只是游离有 html 文档外的节点, 要调用 appendChild 方法才能添加到文档树中
(2) 如果复制的元素有 id, 则其副本同样会包含该 id, 由于 id 具有唯一性, 所以在复制节点后必须要修改其 id
(3) 调用接收的 bool 参数最好传入, 如果不传入该参数, 不同浏览器对其默认值的处理可能不同
除此之外, 我们还有一个需要注意的点:
如果被复制的节点绑定了事件, 则副本也会跟着绑定该事件吗? 这里要分情况讨论:
(1) 如果是通过 addEventListener 或者比如 onclick 进行绑定事件, 则副本节点不会绑定该事件
(2) 如果是内联方式绑定比如
<div onclick="showParent()"></div>
这样的话, 副本节点同样会触发事件
createDocumentFragment
createDocumentFragment 方法用来创建一个 DocumentFragment 在前面我们说到 DocumentFragment 表示一种轻量级的文档, 它的作用主要是存储临时的节点用来准备添加到文档中
createDocumentFragment 方法主要是用于添加大量节点到文档中时会使用到假设要循环一组数据, 然后创建多个节点添加到文档中, 比如示例
- <ul id="list"></ul>
- <input type="button" value="添加多项" id="btnAdd" />
- document.getElementById("btnAdd").onclick = function(){
- var list = document.getElementById("list");
- for(var i = 0;i < 100; i++){
- var li = document.createElement("li");
- li.textContent = i;
- list.appendChild(li);
- }
- }
这段代码将按钮绑定了一个事件, 这个事件创建了 100 个 li 节点, 然后依次将其添加 HTML 文档中这样做有一个缺点: 每次一创建一个新的元素, 然后添加到文档树中, 这个过程会造成浏览器的回流所谓回流简单说就是指元素大小和位置会被重新计算, 如果添加的元素太多, 会造成性能问题这个时候, 就是使用 createDocumentFragment 了
DocumentFragment 不是文档树的一部分, 它是保存在内存中的, 所以不会造成回流问题我们修改上面的代码如下:
- document.getElementById("btnAdd").onclick = function() {
- var list = document.getElementById("list");
- var fragment = document.createDocumentFragment();
- for (var i = 0; i < 100; i++) {
- var li = document.createElement("li");
- li.textContent = i;
- fragment.appendChild(li);
- }
- list.appendChild(fragment);
- }
优化后的代码主要是创建了一个 fragment, 每次生成的 li 节点先添加到 fragment, 最后一次性添加到 list, 大家可以看示例
创建型 API 总结
创建型 api 主要包括 createElement,createTextNode,cloneNode 和 createDocumentFragment 四个方法, 需要注意下面几点:
(1) 它们创建的节点只是一个孤立的节点, 要通过 appendChild 添加到文档中
(2)cloneNode 要注意如果被复制的节点是否包含子节点以及事件绑定等问题
(3) 使用 createDocumentFragment 来解决添加大量节点时的性能问题
页面修改型 API
前面我们提到创建型 api, 它们只是创建节点, 并没有真正修改到页面内容, 而是要调用 appendChild 来将其添加到文档树中我在这里将这类会修改到页面内容归为一类
修改页面内容的 api 主要包括: appendChild,insertBefore,removeChild,replaceChild
appendChild
appendChild 我们在前面已经用到多次, 就是将指定的节点添加到调用该方法的节点的子元素的末尾调用方法如下:
parent.appendChild(child);
child 节点将会作为 parent 节点的最后一个子节点
appendChild 这个方法很简单, 但是还有有一点需要注意: 如果被添加的节点是一个页面中存在的节点, 则执行后这个节点将会添加到指定位置, 其原本所在的位置将移除该节点, 也就是说不会同时存在两个该节点在页面上, 相当于把这个节点移动到另一个地方我们来看例子
<div id="child">
要被添加的节点
- </div>
- <br/>
- <br/>
- <br/>
- <div id="parent">
要移动的位置
- </div>
- <input id="btnMove" type="button" value="移动节点" />
- document.getElementById("btnMove").onclick = function(){
- var child = document.getElementById("child");
- document.getElementById("parent").appendChild(child);
- }
这段代码主要是获取页面上的 child 节点, 然后添加到指定位置, 可以看到原本的 child 节点被移动到 parent 中了
这里还有一个要注意的点: 如果 child 绑定了事件, 被移动时, 它依然绑定着该事件
insertBefore
insertBefore 用来添加一个节点到一个参照节点之前, 用法如下:
parentNode.insertBefore(newNode, refNode);
parentNode 表示新节点被添加后的父节点
newNode 表示要添加的节点
refNode 表示参照节点, 新节点会添加到这个节点之前
我们来看这个例子
<div id="parent">
父节点
<div id="child">
子元素
- </div>
- </div>
- <input type="button" id="insertNode" value="插入节点" />
- var parent = document.getElementById("parent");
- var child = document.getElementById("child");
- document.getElementById("insertNode").onclick = function(){
- var newNode = document.createElement("div");
- newNode.textContent = "新节点"
- parent.insertBefore(newNode,child);
- }
这段代码创建了一个新节点, 然后添加到 child 节点之前
和 appendChild 一样, 如果插入的节点是页面上的节点, 则会移动该节点到指定位置, 并且保留其绑定的事件
关于第二个参数参照节点还有几个注意的地方:
(1)refNode 是必传的, 如果不传该参数会报错
(2) 如果 refNode 是 undefined 或 null, 则 insertBefore 会将节点添加到子元素的末尾
removeChild
removeChild 顾名思义, 就是删除指定的子节点并返回, 用法如下:
var deletedChild = parent.removeChild(node);
deletedChild 指向被删除节点的引用, 它等于 node, 被删除的节点仍然存在于内存中, 可以对其进行下一步操作
注意: 如果被删除的节点不是其子节点, 则程序将会报错我们可以通过下面的方式来确保可以删除:
- if (node.parentNode) {
- node.parentNode.removeChild(node);
- }
通过节点自己获取节点的父节点, 然后将自身删除
replaceChild
replaceChild 用于使用一个节点替换另一个节点, 用法如下
parent.replaceChild(newChild, oldChild);
newChild 是替换的节点, 可以是新的节点, 也可以是页面上的节点, 如果是页面上的节点, 则其将被转移到新的位置
oldChild 是被替换的节点
页面修改型 API 总结
页面修改型 api 主要是这四个接口, 要注意几个特点:
(1) 不管是新增还是替换节点, 如果新增或替换的节点是原本存在页面上的, 则其原来位置的节点将被移除, 也就是说同一个节点不能存在于页面的多个位置
(2) 节点本身绑定的事件会不会消失, 会一直保留着
节点查询型 API
节点查询型 API 也是非常常用的 api, 下面我们分别说明一下每一个 api 的使用
document.getElementById
这个接口很简单, 根据元素 id 返回元素, 返回值是 Element 类型, 如果不存在该元素, 则返回 null
使用这个接口有几点要注意:
(1) 元素的 Id 是大小写敏感的, 一定要写对元素的 id
(2)HTML 文档中可能存在多个 id 相同的元素, 则返回第一个元素
(3) 只从文档中进行搜索元素, 如果创建了一个元素并指定 id, 但并没有添加到文档中, 则这个元素是不会被查找到的
document.getElementsByTagName
这个接口根据元素标签名获取元素, 返回一个即时的 HTMLCollection 类型, 什么是即时的 HTMLCollection 类型呢? 我们来看看这个示例
- <div>
- div1
- </div>
- <div>
- div2
- </div>
- <input type="button" value="显示数量" id="btnShowCount" />
- <input type="button" value="新增 div" id="btnAddDiv" />
- var divList = document.getElementsByTagName("div"); document.getElementById("btnAddDiv").onclick
- = function(){ var div = document.createElement("div"); div.textContent
- ="div" + (divList.length+1); document.body.appendChild(div); } document.getElementById("btnShowCount").onclick
- = function(){ alert(divList.length); }
这段代码中有两个按钮, 一个按钮是显示 HTMLCollection 元素的个数, 另一个按钮可以新增一个 div 标签到文档中前面提到 HTMLCollcetion 元素是即时的表示该集合是随时变化的, 也就是是文档中有几个 div, 它会随时进行变化, 当我们新增一个 div 后, 再访问 HTMLCollection 时, 就会包含这个新增的 div
使用 document.getElementsByTagName 这个方法有几点要注意:
(1) 如果要对 HTMLCollection 集合进行循环操作, 最好将其长度缓存起来, 因为每次循环都会去计算长度, 暂时缓存起来可以提高效率
(2) 如果没有存在指定的标签, 该接口返回的不是 null, 而是一个空的 HTMLCollection
(3)* 表示所有标签
document.getElementsByName
getElementsByName 主要是通过指定的 name 属性来获取元素, 它返回一个即时的 NodeList 对象
使用这个接口主要要注意几点:
(1) 返回对象是一个即时的 NodeList, 它是随时变化的
(2) 在 HTML 元素中, 并不是所有元素都有 name 属性, 比如 div 是没有 name 属性的, 但是如果强制设置 div 的 name 属性, 它也是可以被查找到的
(3) 在 IE 中, 如果 id 设置成某个值, 然后传入 getElementsByName 的参数值和 id 值一样, 则这个元素是会被找到的, 所以最好不好设置同样的值给 id 和 name
document.getElementsByClassName
这个 API 是根据元素的 class 返回一个即时的 HTMLCollection, 用法如下
var elements = document.getElementsByClassName(names);
这个接口有下面几点要注意:
(1) 返回结果是一个即时的 HTMLCollection, 会随时根据文档结构变化
(2)IE9 以下浏览器不支持
(3) 如果要获取 2 个以上 classname, 可传入多个 classname, 每个用空格相隔, 例如
var elements = document.getElementsByClassName("test1 test2");
document.querySelector 和 document.querySelectorAll
这两个 api 很相似, 通过 CSS 选择器来查找元素, 注意选择器要符合 CSS 选择器的规则
首先来介绍一下 document.querySelector
document.querySelector 返回第一个匹配的元素, 如果没有匹配的元素, 则返回 null
注意, 由于返回的是第一个匹配的元素, 这个 api 使用的深度优先搜索来获取元素我们来看这个例子:
- <div>
- <div>
- <span class="test">
第三级的 span
- </span>
- </div>
- </div>
- <div class="test">
同级的第二个 div
- </div>
- <input type="button" id="btnGet" value="获取 test 元素" />
- document.getElementById("btnGet").addEventListener("click",function(){
- var element = document.querySelector(".test"); alert(element.textContent);
- })
这个例子很简单, 就是两个 class 都包含 test 的元素, 一个在文档树的前面, 但是它在第三级, 另一个在文档树的后面, 但它在第一级, 通过 querySelector 获取元素时, 它通过深度优先搜索, 拿到文档树前面的第三级的元素
document.querySelectorAll 的不同之处在于它返回的是所有匹配的元素, 而且可以匹配多个选择符, 我们来看看下面这个例子
<div class="test">
class 为 test
- </div>
- <div id="test">
id 为 test
- </div>
- <input id="btnShow" type="button" value="显示内容" />
- document.getElementById("btnShow").addEventListener("click",function(){
- var elements = document.querySelectorAll("#test,.test");
- for(var i = 0,length = elements.length;i<length;i++){
- alert(elements[i].textContent);
- }
- })
这段代码通过 querySelectorAll, 使用 id 选择器和 class 选择器选择了两个元素, 并依次输出其内容要注意两点:
(1)querySelectorAll 也是通过深度优先搜索, 搜索的元素顺序和选择器的顺序无关
(2) 返回的是一个非即时的 NodeList, 也就是说结果不会随着文档树的变化而变化
兼容性问题: querySelector 和 querySelectorAll 在 ie8 以下的浏览器不支持
节点关系型 api
在 html 文档中的每个节点之间的关系都可以看成是家谱关系, 包含父子关系, 兄弟关系等等, 下面我们依次来看看每一种关系
父关系型 api
parentNode: 每个节点都有一个 parentNode 属性, 它表示元素的父节点 Element 的父节点可能是 Element,Document 或 DocumentFragment
parentElement: 返回元素的父元素节点, 与 parentNode 的区别在于, 其父节点必须是一个 Element, 如果不是, 则返回 null
兄弟关系型 api
previousSibling: 节点的前一个节点, 如果该节点是第一个节点, 则为 null 注意有可能拿到的节点是文本节点或注释节点, 与预期的不符, 要进行处理一下
previousElementSibling: 返回前一个元素节点, 前一个节点必须是 Element, 注意 IE9 以下浏览器不支持
nextSibling: 节点的后一个节点, 如果该节点是最后一个节点, 则为 null 注意有可能拿到的节点是文本节点, 与预期的不符, 要进行处理一下
nextElementSibling: 返回后一个元素节点, 后一个节点必须是 Element, 注意 IE9 以下浏览器不支持
子关系型 api
childNodes: 返回一个即时的 NodeList, 表示元素的子节点列表, 子节点可能会包含文本节点, 注释节点等
children: 一个即时的 HTMLCollection, 子节点都是 Element,IE9 以下浏览器不支持
firstNode: 第一个子节点
lastNode: 最后一个子节点
hasChildNodes 方法: 可以用来判断是否包含子节点
元素属性型 api
setAttribute
setAttribute: 根据名称和值修改元素的特性, 用法如下
element.setAttribute(name, value);
其中 name 是特性名, value 是特性值如果元素不包含该特性, 则会创建该特性并赋值
如果元素本身包含指定的特性名为属性, 则可以世界访问属性进行赋值, 比如下面两条代码是等价的:
- element.setAttribute("id", "test");
- element.id = "test";
- getAttribute
getAttribute 返回指定的特性名相应的特性值, 如果不存在, 则返回 null 或空字符串用法如下:
var value = element.getAttribute("id");
元素样式型 api
window.getComputedStyle
window.getComputedStyle 是用来获取应用到元素后的样式, 假设某个元素并未设置高度而是通过其内容将其高度撑开, 这时候要获取它的高度就要用到 getComputedStyle, 用法如下:
var style = window.getComputedStyle(element[, pseudoElt]);
element 是要获取的元素, pseudoElt 指定一个伪元素进行匹配
返回的 style 是一个 CSSStyleDeclaration 对象
通过 style 可以访问到元素计算后的样式
getBoundingClientRect
getBoundingClientRect 用来返回元素的大小以及相对于浏览器可视窗口的位置, 用法如下:
var clientRect = element.getBoundingClientRect();
clientRect 是一个 DOMRect 对象, 包含 left,top,right,bottom, 它是相对于可视窗口的距离, 滚动位置发生改变时, 它们的值是会发生变化的除了 IE9 以下浏览器, 还包含元素的 height 和 width 等数据, 具体可查看链接
总结
本文主要总结了原生 js 中常用的操作 DOM 的 api 接口, 主要为了复习基础知识平时开发用多了 jQuery 等类库, 对基础知识的了解可能就渐渐地遗忘, 但这些基础知识才是我们立足的根本, 只有掌握原生的 js, 才能真正做好 js 的开发
来源: http://www.codeceo.com/article/javascript-dom-api.html