Vue中fragment.js使用方法小結(jié)
createDocumentFragment
如果要在一個(gè)節(jié)點(diǎn)上一次性插入多個(gè)元素怎么辦,比如說(shuō)一次插入 10000 個(gè)節(jié)點(diǎn)?
最簡(jiǎn)單粗暴的方式就是:
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);
}
不過(guò)眾所周知的原因,對(duì) DOM 反復(fù)操作會(huì)導(dǎo)致頁(yè)面重繪、回流,效率非常低,而且頁(yè)面可能會(huì)被卡死,這段代碼基本是沒(méi)人用的。
如果分段來(lái)進(jìn)行 DOM 操作呢,這樣就能避免卡死頁(yè)面了,js 忍者秘籍里面提到過(guò)可以用 setTimeout 來(lái)改進(jìn):
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);
當(dāng)然,更多能想到的方式應(yīng)該是,在內(nèi)存中直接操作節(jié)點(diǎn),所有節(jié)點(diǎn)都湊在一起之后再跟 DOM 樹(shù)進(jìn)行交互,把所有節(jié)點(diǎn)都串在一個(gè) div 上,然后再把 div 掛到 DOM 樹(shù)上:
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 樹(shù)交互一次,性能方面肯定是大有改善的,不過(guò)額外插入了一個(gè) div,如果說(shuō)不是跟div之類(lèi)的節(jié)點(diǎn)進(jìn)行交互呢,比如在 table 中插入 th、td?
這時(shí)候,createDocumentFragment 就該出馬了,翻譯過(guò)來(lái)叫“文檔片段”,按MDN的描述:
DocumentFragments 是一些 DOM 節(jié)點(diǎn)。它們不是 DOM 樹(shù)的一部分。通常的使用場(chǎng)景是創(chuàng)建一個(gè)文檔片段,然后將創(chuàng)建的 DOM 元素插入到文檔片段中,最后把文檔片段插入到 DOM 樹(shù)中。在 DOM 樹(shù)中,文檔片段會(huì)被替換為它所有的子元素。
因?yàn)槲臋n片段存在與內(nèi)存中,并不在 DOM 樹(shù)中,所以將子元素插入到文檔片段時(shí)不會(huì)引起頁(yè)面回流(對(duì)元素位置和幾何上的計(jì)算)。因此,使用文檔片段 document fragments 通常會(huì)起到優(yōu)化性能的作用。
簡(jiǎn)單來(lái)說(shuō),就是上面一個(gè)例子的不需要 div 中轉(zhuǎn)版本,插入的時(shí)候,直接用其子元素替換其本身,非常完美。
雖然說(shuō),“好用的都不通用”(特別是針對(duì)某公司瀏覽器),不過(guò)這個(gè)好用的東西,甚至連 IE6 都支持。
具體代碼大概就長(zhǎng)這樣:
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);
具體性能方面的測(cè)試,有興趣的可以把所有代碼都跑一遍。
innerHTML
把一長(zhǎng)串字符串轉(zhuǎn)換為對(duì)應(yīng)的 DOM 節(jié)點(diǎn),正常而言,首先想到的肯定是 innerHTML。大概流程就是,先創(chuàng)建一個(gè) div 節(jié)點(diǎn),然后 div.innerHTML = str,根據(jù)需要把 div 的 children 取出來(lái)放到該放的地方去,div 本身給扔了。
如果想單獨(dú)生成一個(gè) th 節(jié)點(diǎn)呢?
試試上面的流程:
var div = document.createElement(`'div'`); div.innerHTML = '<th>xxx</th>'`;` console.log(div);
實(shí)際輸出是(chrome 下):
<`div>xxx</div`>
并沒(méi)有得到想要的:
<`div><th>xxx</th></div`>
對(duì)于這樣的結(jié)果是可以理解的,畢竟一個(gè) th 放到 div 里面,怎么看都不對(duì),直接把外圍的標(biāo)簽去掉,內(nèi)容扔到 div 里面也是相當(dāng)智能的。
不過(guò)架不住,有時(shí)候就是要獲取一個(gè) th 節(jié)點(diǎn)。
其實(shí)也好辦,寫(xiě)全了不就得了:
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);
可以看出,結(jié)果正是所想要的。
fragment.js
// 需要單獨(dú)處理的一些特殊節(jié)點(diǎn)
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');
// 拼接節(jié)點(diǎn)字符串
node.innerHTML = prefix + templateString.trim() + suffix;
// 去除外包裹層,只留字符串轉(zhuǎn)化的節(jié)點(diǎn)
while(depth--) node = node.lastChild;
// 只有一個(gè)節(jié)點(diǎn)的情況
if(node.firstChild === node.lastChild) {
frag.appendChild(node.firstChild);
return frag;
}
// 多個(gè)節(jié)點(diǎn),依序添加到 frag
var child;
while(child = node.firstChild) {
frag.appendChild(child);
}
return frag;
}
相關(guān)文章
vue+Element中table表格實(shí)現(xiàn)可編輯(select下拉框)
這篇文章主要介紹了vue+Element中table表格實(shí)現(xiàn)可編輯,實(shí)現(xiàn)select下拉框,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-05-05
解決Vue打包上線(xiàn)之后部分CSS不生效的問(wèn)題
今天小編就為大家分享一篇解決Vue打包上線(xiàn)之后部分CSS不生效的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-11-11
解決Vue在封裝了Axios后手動(dòng)刷新頁(yè)面攔截器無(wú)效的問(wèn)題
這篇文章主要介紹了解決VUE在封裝了Axios后手動(dòng)刷新頁(yè)面攔截器無(wú)效的問(wèn)題,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2018-11-11
vue跳轉(zhuǎn)同一個(gè)組件,參數(shù)不同,頁(yè)面接收值只接收一次的解決方法
今天小編就為大家分享一篇vue跳轉(zhuǎn)同一個(gè)組件,參數(shù)不同,頁(yè)面接收值只接收一次的解決方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-11-11
vue實(shí)現(xiàn)鼠標(biāo)移入移出事件代碼實(shí)例
這篇文章主要介紹了vue實(shí)現(xiàn)鼠標(biāo)移入移出事件,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-03-03
vue中常見(jiàn)的問(wèn)題及解決方法總結(jié)(推薦)
這篇文章主要給大家介紹了關(guān)于vue中常見(jiàn)的問(wèn)題及解決方法的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-04-04
Vue數(shù)據(jù)監(jiān)聽(tīng)器watch和watchEffect的使用
今天我們來(lái)學(xué)習(xí)一下watch監(jiān)聽(tīng)器和它的好兄弟watchEffect監(jiān)聽(tīng)器。這個(gè)相對(duì)來(lái)說(shuō)比較簡(jiǎn)單,用的不是很多,當(dāng)然了,根據(jù)自己的項(xiàng)目情況自行決定使用,希望對(duì)大家有所幫助2023-02-02

