3操作节点

  因为关系节点都是只读的,所以DOM提供了一些操作节点的方法.其中,最常用的方法是appendChild().用于向childNodes列表的末尾添加一个节点,添加节点后,childNodes的新增节点,父节点及以前的最后一个子节点的关系指针都会相应地得到更新.更新完成后,appendChild()返回新增的节点.

  来看下面的例子:

  var returnNode = someNode.appendChild(newNode);

  alert( returnNode==newNode ); //true

  alert(someNode.lastChild==newNode); //true;

  如果传入到appendChild()中的节点已经是文档的一部分了,那结果就是将该节点从原来的位置转移到新位置.即使可以将DOM树看成是由一系列指针连接起来的,但任何DOM节点也不能同时出现在文档树中的多个位置上.因此,如果在调用appendChild()时传入了父节点的第一个子节点,那么该节点就会成为父节点的最后一个子节点.

  如果需要把节点放在childNodes列表中某个特定的位置上,而不是放在末尾,那么可以使用insertBefore()方法,这个方法接受两个参数,要插入的节点和作为参照的节点,插入节点后,被插入的节点会变成参照节点的前一个同胞节点(previousSibling),同时被方法返回,如果参照节点是null,则insertBefore()与appendChild()执行相同的操作:

 1     //插入后成为最后一个子节点
 2 
 3     returnedNode = someNode.insertBefore(newNode,null);
 4 
 5     alert( newNode == someNode.lastChild ); //true
 6 
 7 
 8   //插入后成为第一个子节点
 9 
10   var  returnedNode=someNode.insertBefore(newNode,someNode.firstChild);
11 
12   alert( returnedNode==newNode );//true
13 
14   alert( newNode==someNode.firstChild );
15 
16 
17     //插入到最后一个子节点前面
18 
19     returnedNode=someNode.insertBefore(newNode,someNode.lastChild);
20     alert(newNode=someNode,childNodes[someNode.childNodes.length-2]);//true
21 
22 //上面介绍的appendChild()和insertBefore()方法都只插入节点,不会移除节点.
23  

replaceChild()方法接受两个参数:要插入的节点和要替换的节点,要替换的节点将由这个方法返回并从文档树中被移除,同时由要插入的节点占据其位置:

1 //替换第一个子节点
2 
3 var returnedNode=someNode.replaceChild(newNode,someNode.firstChild);
4 
5  
6 //替换最后一个子节点
7 
8 returnedNode=someNode.replaceChild(newNode,someNode.lastChild);

 

在使用replaceChild()插入一个节点时,该节点的所有关系指针都会从被它替换的节点复制过来.尽管从技术上讲,被替换的节点仍然还在文档中,但它在文档中已经没有了自己的位置.

如果只想移除而非替换节点,可以使用removeChild()方法,这个方法接收一个参数,既要被移除的节点,被移除的节点将成为方法的返回值:

1 //移除第一个子节点
2 
3 var formerFirstChild=someNode.removeChild(someNode.firstChild);
4 
5  
6 
7 移除最后一个子节点
8 
9 var formerLastChild=someNode.removeChild(someNode.lastChild);

 

4其他方法

  有两个方法是所有类型节点都有的,第一个就是cloneNode(),用于创建调用这个方法的节点的一个完全相同的副本.cloneNode()方法接受一个布尔值参数,表示是否执行深复制,true表示深复制,也就是复制节点及其整个子节点数;false为浅复制,即只复制节点本身.复制后,返回的节点副本属于文档所有,但并没有为它制定父节点.因此,这个节点副本就成为了一个孤儿,除非通过appendChild(),insertBefore()或replaceChild()将它添加到文档中:

<ul>

<li>1</li>

<li>2</li>

<li>3</li>

</ul>

如果已经将<ul>元素的引用保存在了变量myList中,那么通常下列代码就可以看书使用cloneNode()方法的两种模式.

var deepList=myList.cloneNode(true);

alert(deepList.childNodes.length);//3(ie<9)或7(其它浏览器)

 

var shallowList=myList.cloneNode(false);

alert(shallowList.childNodes.length);//0

  在这个例子中,deepList中保存着一个对myList执行深复制得到的副本.因此,deepList中包含三个列表项,每个列表项中都包含文本.而变量在shallowList中保存着对myList执行浅复制得到的副本,因此它不包含子节点.deepList.childNodes.length中的差异主要是因为IE8及更早版本与其他浏览器处理空白字符的方式不一样IE9之前的版本不会为空白字符创建节点.

  cloneNode()方法不会复制添加到DOM节点中的JavaScript属性,例如事件处理程序等,这个方法只复制特性,子节点,其他一切都不会复制,ie在此存在一个bug,即它会复制事件处理程序,所以,建议在复制之前最好移除事件处理程序.

 posted on 2015-05-11 00:07  gupeng1988  阅读(96)  评论(0)    收藏  举报