大部分内容源自 jQuery,当然,同时也参考了 component/domify ,如果有兴趣去这翻阅原始的代码,可以到 jQuery 中查找 wrapMap;至于 domify,直接到 github 搜索即可,相关项目类容很少,直接看 index.js 就行了。
createDocumentFragment
如果要在一个节点上一次性插入多个元素怎么办,比如说一次插入 10000 个节点?
最简单粗暴的方式就是:
var parent = document.getElementById('parent'); for(var i = 0; i < 10000; i++) { var child = document.createElement('div'); var text = document.createTextNode('' + i); child.appendChild(text); parent.appendChild(child); }
不过众所周知的原因,对 DOM 反复操作会导致页面重绘、回流,效率非常低,而且页面可能会被卡死,这段代码基本是没人用的。
如果分段来进行 DOM 操作呢,这样就能避免卡死页面了,js 忍者秘籍里面提到过可以用 setTimeout 来改进:
var i = 0, max = 10000; setTimeout(function addNodes() { for(var step = i + 500; i < step; i++) { var child = document.createElement('div'); child.appendChild(document.createTextNode('' + i)); div.appendChild(child); } if(i < max) { setTimeout(addNodes, 0); } }, 0);
当然,更多能想到的方式应该是,在内存中直接操作节点,所有节点都凑在一起之后再跟 DOM 树进行交互,把所有节点都串在一个 div 上,然后再把 div 挂到 DOM 树上:
var parent = document.getElementById('parent'); var div = document.createElement('div'); for(var i = 0; i < 10000; i++) { var child = document.createElement('div'); var text = document.createTextNode('' + i); child.appendChild(text); div.appendChild(child); } parent.appendChild(div);
如上,只跟 DOM 树交互一次,性能方面肯定是大有改善的,不过额外插入了一个 div,如果说不是跟div之类的节点进行交互呢,比如在 table 中插入 th、td?
这时候,createDocumentFragment 就该出马了,翻译过来叫“文档片段”,按MDN的描述:
DocumentFragments 是一些 DOM 节点。它们不是 DOM 树的一部分。通常的使用场景是创建一个文档片段,然后将创建的 DOM 元素插入到文档片段中,最后把文档片段插入到 DOM 树中。在 DOM 树中,文档片段会被替换为它所有的子元素。
因为文档片段存在与内存中,并不在 DOM 树中,所以将子元素插入到文档片段时不会引起页面回流(对元素位置和几何上的计算)。因此,使用文档片段 document fragments 通常会起到优化性能的作用。
简单来说,就是上面一个例子的不需要 div 中转版本,插入的时候,直接用其子元素替换其本身,非常完美。
虽然说,“好用的都不通用”(特别是针对某公司浏览器),不过这个好用的东西,甚至连 IE6 都支持。
具体代码大概就长这样:
var parent = document.getElementById('parent'); var frag = document.createDocumentFragment(); for(var i = 0; i < 10000; i++) { var child = document.createElement('div'); var text = document.createTextNode('' + i); child.appendChild(text); frag.appendChild(child); } parent.appendChild(frag);
具体性能方面的测试,有兴趣的可以把所有代码都跑一遍。
innerHTML
把一长串字符串转换为对应的 DOM 节点,正常而言,首先想到的肯定是 innerHTML。大概流程就是,先创建一个 div 节点,然后 div.innerHTML = str,根据需要把 div 的 children 取出来放到该放的地方去,div 本身给扔了。
如果想单独生成一个 th 节点呢?
试试上面的流程:
var div = document.createElement('div'); div.innerHTML = '<th>xxx</th>'; console.log(div);
实际输出是(chrome 下):
<div>xxx</div>
并没有得到想要的:
<div><th>xxx</th></div>
对于这样的结果是可以理解的,毕竟一个 th 放到 div 里面,怎么看都不对,直接把外围的标签去掉,内容扔到 div 里面也是相当智能的。
不过架不住,有时候就是要获取一个 th 节点。
其实也好办,写全了不就得了:
var node = document.createElement('div'); node.innerHTML = '<table><tbody><tr><th>xxx</th></tr></tbody></table>'; // 把外面的几层皮扒掉就是想要的 th 了 var depth = 3; while(depth--) { node = node.lastChild; } console.log(node.firstChild);
可以看出,结果正是所想要的。
fragment.js
// 需要单独处理的一些特殊节点 var map = { legend : [1, '<fieldset>', '</fieldset>'], tr : [2, '<table><tbody>', '</tbody></table>'], col : [2, '<table><tbody></tbody><colgroup>', '</colgroup></table>'], _default : [0, '', ''] }; map.td = map.th = [3, '<table><tbody><tr>', '</tr></tbody></table>']; map.option = map.optgroup = [1, '<select multiple="multiple">', '</select>']; map.thead = map.tbody = map.colgroup = map.caption = map.tfoot = [1, '<table>', '</table>'] map.text = map.circle = map.ellipse = map.line = map.path = map.polygon = map.polyline = map.rect = [1, '<svg xmlns="http://www.w3.org/2000/svg" version="1.1">','</svg>']; var TAG_RE = /<([\w:]+)/; module.exports = function(templateString) { var frag = document.createDocumentFragment(), m = TAG_RE.exec(templateString); // 单纯字符串的情况 if(!m) { frag.appendChild(document.createTextNode(templateString); return frag; } var tag = m[1], wrap = map[tag] || map._default, depth = wrap[0], prefix = wrap[1], suffix = wrap[2], node = document.createElement('div'); // 拼接节点字符串 node.innerHTML = prefix + templateString.trim() + suffix; // 去除外包裹层,只留字符串转化的节点 while(depth--) node = node.lastChild; // 只有一个节点的情况 if(node.firstChild === node.lastChild) { frag.appendChild(node.firstChild); return frag; } // 多个节点,依序添加到 frag var child; while(child = node.firstChild) { frag.appendChild(child); } return frag; }
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持小牛知识库。
本文向大家介绍Vue中fragment.js使用方法小结,包括了Vue中fragment.js使用方法小结的使用技巧和注意事项,需要的朋友参考一下 createDocumentFragment 如果要在一个节点上一次性插入多个元素怎么办,比如说一次插入 10000 个节点? 最简单粗暴的方式就是: 不过众所周知的原因,对 DOM 反复操作会导致页面重绘、回流,效率非常低,而且页面可能会被卡死,这段
本文向大家介绍vue component组件使用方法详解,包括了vue component组件使用方法详解的使用技巧和注意事项,需要的朋友参考一下 什么是组件 按照惯例,引用Vue官网的一句话: 组件 (Component) 是 Vue.js 最强大的功能之一。组件可以扩展 HTML 元素,封装可重用的代码。在较高层面上,组件是自定义元素,Vue.js 的编译器为它添加特殊功能。在有些情况下,组件
本文向大家介绍详解vue slot插槽的使用方法,包括了详解vue slot插槽的使用方法的使用技巧和注意事项,需要的朋友参考一下 官方文档其实已经讲得很详细,我根据文档,把官方的小案例实现了一下,这样更直观 单个slot使用最简单,也是最常用的,当我们定义了一个子组件,父组件在使用的这个组件的时候,想在内部自定义一些初始化数据,这时候就可以用slot实现。 具名slot只是给slot加了name
本文向大家介绍Vue.js路由vue-router使用方法详解,包括了Vue.js路由vue-router使用方法详解的使用技巧和注意事项,需要的朋友参考一下 vue-router是Vue.js官方的路由插件,它和vue.js是深度集成的,适合用于构建单页面应用。vue的单页面应用是基于路由和组件的,路由用于设定访问路径,并将路径和组件映射起来。传统的页面应用,是用一些超链接来实现页面切换和跳转的
本文向大家介绍vue拖拽组件使用方法详解,包括了vue拖拽组件使用方法详解的使用技巧和注意事项,需要的朋友参考一下 前言 pc端开发需要拖拽组件完成列表的顺序交换,一般移动端的UI组件会包含,但是我在用的iview并没有此功能的组件,于是手写一个,实现起来很简单。效果图如下: 可以拖拽完成新排序,点击某一项可以触发相关事件. 关于拖拽 drag & drop 拖放(Drag 和 drop)是 HT
本文向大家介绍详解vue 中使用 AJAX获取数据的方法,包括了详解vue 中使用 AJAX获取数据的方法的使用技巧和注意事项,需要的朋友参考一下 在VUE开发时,数据可以使用jquery和vue-resource来获取数据。在获取数据时,一定需要给一个数据初始值。 看下例: 这里必须设置 vue的data的初始数据,即使此时数据为空。 在使用ajax获取数据时,使用vue-resource 更加