JS魔法堂:元素克隆、剪切技术研究
一、前言
当需要新元素时我们可以通过 document.createElement 接口来创建一个全新的元素,也可以通过克隆已有元素的方式来获取一个新元素。而在部分浏览器中,通过复制来获取新元素的效率比通过 document.createElement 方式的要高一些,具体的性能比较如下:
2% in IE8, but no change in IE6 and IE7
Up to 5.5% in Firefox 3.5 and Safari 4
6% in Opera (but no savings in Opera 10)
10% in Chrome 2 and 3% in Chrome 3
本篇将记录元素克隆、和剪切的相关技术,以便日后查阅。
目录一坨:
2. appendChild、insertBefore和replaceChild
五、题外话——IE独有的replaceNode和swapNode方法
二、拷贝
1、Node.cloneNode
浏览器支持:所有浏览器均支持。
作用:拷贝元素自身。
API规范: {Node} Node.clone({boolean} [isDeep=false]) ,默认情况下仅拷贝元素本身,若入参为true时拷贝子孙元素也将被一同拷贝。
实际测试效果:
| 浏览器 | 复制子元素 | 标准属性(property) | 标准特性(attribute) | 自定义特性(customize attribute) | 自定义属性(expando) | DOM0事件处理函数 | DOM2事件处理函数 |
parentNode和 parentElement的值 |
ownerDocument |
| IE5.5~8 | √ | √ | √ | √ | √(浅复制) | Χ | Χ | null | 不变 |
| IE9+ | √ | √ | √ | √ | Χ | Χ | Χ | null | 不变 |
| Chrome | √ | √ | √ | √ | Χ | Χ | Χ | null | 不变 |
| FF | √ | √ | √ | √ | Χ | Χ | Χ | null | 不变 |
注意:
1. 使用cloneNode会将id特性也复制,因此需要手动修改副本的id特性。
2. {Document} document和{HTMLDocument} document.documentElement也可以调用cloneNode方法拷贝自身,并且支持深拷贝。
3. 当从其他文档中拷贝元素,元素副本的ownerDocument依然为其他文档的document对象,直到我们将元素副本添加到当前文档下,ownerDocument属性才会变化。
2、document.importNode
浏览器支持:IE9+和其他现代浏览器均支持。
作用:拷贝其他文档中的元素到当前文档中。(https://developer.mozilla.org/en-US/docs/Web/API/document.importNode)
API规范: {Node} document.importNode({HTMLElement|HTMLDocument|HTMLDocumentFragment} externalNode [, {boolean} isDeep=true])
实际测试效果:
| 浏览器 | 复制子元素 | 标准属性(property) | 标准特性(attribute) | 自定义特性(customize attribute) | 自定义属性(expando) | DOM0事件处理函数 | DOM2事件处理函数 |
parentNode和 parentElement的值 |
ownerDocument |
| IE9+ | √ | √ | √ | √ | Χ | Χ | Χ | null | 当前文档的document对象 |
| Chrome | √ | √ | √ | √ | Χ | Χ | Χ | null | 当前文档的document对象 |
| FF | √ | √ | √ | √ | Χ | Χ | Χ |
null | 当前文档的document对象 |
注意:
1. 使用importNode会将id特性也复制,因此需要手动修改副本的id特性;
2. 不接受{Document} document的拷贝;
3. 虽然规范中描述其作用为拷贝其他文档中的元素,但实际上是可以对当前文档的元素进行拷贝的;
4. 当从其他文档中拷贝元素,元素副本的ownerDocument自动设置为当前文档的document对象。
三、剪切
1、document.adoptNode
浏览器支持:IE9+和其他现代浏览器均支持。
作用:剪切其他文档中的元素到当前文档中。(https://developer.mozilla.org/en-US/docs/Web/API/document.adoptNode)
API规范: {Node} document.adoptNode({HTMLElement|HTMLDocument|HTMLDocumentFragment} externalNode)
实际测试效果:
| 浏览器 | 复制子元素 | 标准属性(property) | 标准特性(attribute) | 自定义特性(customize attribute) | 自定义属性(expando) | DOM0事件处理函数 | DOM2事件处理函数 |
parentNode和 parentElement的值 |
ownerDocument |
| IE9+ | √ | √ | √ | √ | √ | √ | √ | null | 当前文档的document对象 |
| Chrome | √ | √ | √ | √ | √ | √ | √ | null | 当前文档的document对象 |
| FF | √ | √ | √ | √ | √ | √ | √ | null | 当前文档的document对象 |
注意:
1. 不接受{Document} document的剪切,但可以对{HTMLDocument} document.documentElement进行剪切;
2. 虽然规范中描述其作用为拷贝其他文档中的元素,但实际上是可以对当前文档的元素进行拷贝的;
3. 当从其他文档中拷贝元素,元素副本的ownerDocument自动设置为当前文档的document对象。
2. appendChild、insertBefore和replaceChild
我们知道appendChild、insertBefore和replaceChild操作元素时会切断元素原来的位置关系,然后将其添加到新的树层级结构中。这不就是元素的剪切操作吗!于是我们可以通过appendChild、insertBefore和replaceChild方法将目标元素剪切到一个未加入DOM树的元素中,即可模拟document.adoptNode的功能了。
;(function(doc){
var clipboard
doc.adoptNode = doc.adoptNode || (clipboard = document.createElement('div'), function(node){
clipboard.appendChild(node)
return clipboard.lastChild
})
}(document))
实际测试效果:
| 浏览器 | 复制子元素 | 标准属性(property) | 标准特性(attribute) | 自定义特性(customize attribute) | 自定义属性(expando) | DOM0事件处理函数 | DOM2事件处理函数 |
parentNode和 parentElement的值 |
ownerDocument |
| IE9+ | √ | √ | √ | √ | √ | √ | √ | 充当
clipboard |
当前文档的document对象 |
| Chrome | √ | √ | √ | √ | √ | √ | √ | 充当
clipboard |
当前文档的document对象 |
| FF | √ | √ | √ | √ | √ | √ | √ | 充当
clipboard |
当前文档的document对象 |
注意:
1. 不接受{Document} document和{HTMLDocument} document.documentElement的剪切,但可以对{HTMLBodyElement} document.body进行剪切;
2. 当从其他文档中拷贝元素,元素副本的ownerDocument自动设置为当前文档的document对象。
四、总结
上述的元素拷贝操作均无法拷贝自定义属性和事件处理绑定,而jQuery的clone函数可实现这一点。
尊重原创,转载请注明来自:http://www.cnblogs.com/fsjohnhuang/p/4176612.html ^_^肥子John
五、题外话——IE独有的replaceNode和swapNode方法
IE5.5~11还提供了 el.replaceNode({HTMLElement} otherEl) 和 el.swapNode(HTMLElement} otherEl) 两个方法, el.replaceNode({HTMLElement} otherEl) 作用是将el替换为otherEl并将el作为函数返回值, 此时el已经脱离了DOM树; el.swapNode(HTMLElement} otherEl) 作用是交换el和otherEl在树层级结构中的位置,两者均在DOM树中。 注意:这两个方法均为IE独有的。
JS魔法堂:元素克隆、剪切技术研究的更多相关文章
- JS魔法堂:LINK元素深入详解
一.前言 我们一般使用方式为 <link type="text/css" rel="stylesheet" href="text.css&quo ...
- JS魔法堂:IMG元素加载行为详解
一.前言 在<JS魔法堂:jsDeferred源码剖析>中我们了解到img元素加载失败可以作为函数异步执行的优化方案,本文打算对img元素的加载行为进行更深入的探讨. 二.资源加载的相关属 ...
- JS魔法堂:jsDeferred源码剖析
一.前言 最近在研究Promises/A+规范及实现,而Promise/A+规范的制定则很大程度地参考了由日本geek cho45发起的jsDeferred项目(<JavaScript框架设计& ...
- JS魔法堂:不完全国际化&本地化手册 之 理論篇
前言 最近加入到新项目组负责前端技术预研和选型,其中涉及到一个熟悉又陌生的需求--国际化&本地化.熟悉的是之前的项目也玩过,陌生的是之前的实现仅仅停留在"有"的阶段而已. ...
- JS魔法堂:不完全国际化&本地化手册 之 实战篇
前言 最近加入到新项目组负责前端技术预研和选型,其中涉及到一个熟悉又陌生的需求--国际化&本地化.熟悉的是之前的项目也玩过,陌生的是之前的实现仅仅停留在"有"的阶段而已. ...
- JS魔法堂:判断节点位置关系
一.前言 在polyfill querySelectorAll 和写弹出窗时都需要判断两个节点间的位置关系,通过jQuery我们可以轻松搞定,但原生JS呢?下面我将整理各种判断方法,以供日后查阅. 二 ...
- JS魔法堂:属性、特性,傻傻分不清楚
一.前言 或许你和我一样都曾经被下面的代码所困扰 var el = document.getElementById('dummy'); el.hello = "test"; con ...
- JS魔法堂:那些困扰你的DOM集合类型
一.前言 大家先看看下面的js,猜猜结果会怎样吧! 可选答案: ①. 获取id属性值为id的节点元素 ②. 抛namedItem is undefined的异常 var nodes = documen ...
- JS魔法堂:doctype我们应该了解的基础知识
一.前言 什么是doctype?其实我们一直使用,却很少停下来看清楚它到底是什么,对网页有什么作用.本篇将和大家一起探讨那个默默无闻的doctype吧! 二.什么是doctype doctype或DT ...
- JS魔法堂:浏览器模式和文档模式怎么玩?
一.前言 从IE8开始引入了文档兼容模式的概念,作为开发人员的我们可以在开发人员工具中通过“浏览器模式”和“文档模式”(IE11开始改为“浏览器模式”改成更贴切的“用户代理字符串”)品味一番,它的出现 ...
随机推荐
- 对InvokeAction简略分析了解验证失败为什么Action还会继续执行
一.前言 有些同学使用AuthorizationFilter来进行用户是否登录验证,如果未登录就跳到登录页. 很简单的一个场景,但是有些同学会发现虽然验证失败了,但是整个Action还会执行一遍. 于 ...
- Http规范
1. 关于 HTTP Basic Authentication http://blog.itpub.net/23071790/viewspace-709367/ 通过以下代码,提示用户登录 Respo ...
- dijit样式定制(二)dijit.form.Select与dijit.form.NumberSpinner
dijit.form.Select: Select的样式位于Claro/form/Select.less中,Select主要通过table来布局,下图可以看到Select的布局结构 介绍几个主要的cl ...
- [.net 面向对象编程基础] (18) 泛型
[.net 面向对象编程基础] (18) 泛型 上一节我们说到了两种数据类型数组和集合,数组是指包含同一类型的多个元素,集合是指.net中提供数据存储和检索的专用类. 数组使用前需要先指定大小,并且检 ...
- HTTP协议解析
1. HTTP版本 HTTP/1.0 HTTP/1.1 HTTP-NG 2. 会话方式 HTTP/1.0 建立连接->请求->响应->断开连接 每次连接只处理一次请求和相应,对资源的 ...
- HTTP学习笔记(五)
目前,市场上流行有很多web服务器软件,每种服务器都有自己的特点.我们在开发的过程中,经常要和它们打交道,所以了解它们的工作原理也是很重要的. 几款比较流行的服务器 它们会做些什么? 第三篇中有这样的 ...
- 可拖动的DIV续
之前写过一篇可拖动的DIV讲如何实现可拖动的元素,最后提出了几点不足,这篇文章主要就是回答着三个问题 1. 浏览器兼容性 2. 边界检查 3. 拖动卡顿.失灵 先附上上次代码 <!DOCTYPE ...
- 说说设计模式~组合模式(Composite)
返回目录 何时能用到它? 组合模式又叫部分-整体模式,在树型结构中,模糊了简单元素和复杂元素的概念,客户程序可以向处理简单元素一样来处理复杂元素,从而使得客户程序与复杂元素的内部结构解耦.对于今天这个 ...
- [C#基础]基础知识一: 面向对象的基本知识.
激励自己有时间多看看.!! C#基础共分为七个部分: 一: 面向对象 二: 值类型, 引用类型, 字符串操作 三: 集合文件操作 四: 正则表达式 五: XML操作 六: 委托, 事件 七: 反射 1 ...
- Java基础-继承-编写一个Java应用程序,设计一个汽车类Vehicle,包含的属性有车轮个数 wheels和车重weight。小车类Car是Vehicle的子类,其中包含的属性有载人数 loader。卡车类Truck是Car类的子类,其中包含的属性有载重量payload。每个 类都有构造方法和输出相关数据的方法。最后,写一个测试类来测试这些类的功 能。
#29.编写一个Java应用程序,设计一个汽车类Vehicle,包含的属性有车轮个数 wheels和车重weight.小车类Car是Vehicle的子类,其中包含的属性有载人数 loader.卡车类T ...