Vue頁面渲染中key的應用實例教程
引言
在前端項目開發(fā)過程中,el-table展示的結果列使用組件形式引入,其中某些字段通過:formatter方法轉(zhuǎn)碼,結果欄位的字段顯示/隱藏控制也使用組件形式引入,前端在控制字段顯示屬性時,發(fā)現(xiàn)碼值轉(zhuǎn)換及字段信息展示均有問題。
問題分析
通過閱讀代碼結構,發(fā)現(xiàn)el-table-column通過template循環(huán)生成,由于template的作用是模板占位符,可幫助我們包裹元素,但在循環(huán)過程當中,template不會被渲染到頁面上。有關表格數(shù)據(jù)渲染中key的作用如下:
- key作為一個DOM節(jié)點的標識值,結合Diff算法可以實現(xiàn)對節(jié)點的復用。(key相同的節(jié)點會被復用);
- 只有當key(或其他導致isSameNode判斷為false)發(fā)生改變時,才會觸發(fā)節(jié)點的重新渲染。否則Vue將會復用之前的節(jié)點,通過改變節(jié)點的屬性來實現(xiàn)節(jié)點的更新。
同時,template標簽不支持:key屬性,
注意: vue實例綁定的元素內(nèi)部的template標簽不支持v-show指令,即v-show="false"對template標簽來說不起作用。但是此時的template標簽支持v-if、v-else-if、v-else、v-for這些指令。
解決方法
既然template標簽不支持key屬性,可通過在el-table-column標簽加入:key="Math.random()"屬性,這個key屬性是vue自帶的特殊屬性,主要用在 Vue 的虛擬 DOM 算法,在新舊 nodes 對比時辨識 VNodes,依次來提升頁面渲染性能。如果不更新這個key的話,顯示/隱藏列的時候,部分DOM不會重新渲染,導致table變化時候內(nèi)容錯亂。
拓展閱讀
一、key的作用
前文已經(jīng)講到,作為一個DOM節(jié)點的標識值,結合Diff算法可實現(xiàn)對節(jié)點的復用。(key相同的節(jié)點會被復用。)
只有當key(或其他導致isSameNode判斷為false)發(fā)生改變時,才會觸發(fā)節(jié)點的重新渲染。否則Vue將會復用之前的節(jié)點,通過改變節(jié)點的屬性來實現(xiàn)節(jié)點的更新。那么,key使用id與index的區(qū)別又是什么呢?
二、key使用id與index的區(qū)別
不推薦使用index作為key,因為這種做法會導致某些節(jié)點被錯誤地原地復用,具體如下:
- 性能損耗:列表渲染時會導致變動項往后的所有列表節(jié)點(內(nèi)容)的更新(相當于key沒發(fā)揮作用)。
- 出現(xiàn)錯誤:某些節(jié)點在錯誤的位置被復用。(例如當列表項中使用到復選框時)
性能損耗
列表渲染時會導致變動項往后的所有列表節(jié)點(內(nèi)容)的更新(相當于key沒發(fā)揮作用)
需要注意的是,變動項往后的所有列表節(jié)點的更新本質(zhì)是節(jié)點屬性的更新,節(jié)點本身會被復用。
<!-- 測試代碼 --> <template> <div> <div v-for="(item, index) in arr" :key="index 或 item.id"> {{item.data}} </div> </div> </template> <script> export default { name: 'HelloWorld', data(){ return { arr: Array.from({length: 10000}, (v, i) => {return {id: i, data: i}}) } }, mounted(){ setTimeout(()=>{ /* 1. this.shiftArr() // 刪除首項 或 2. this.unShiftArr() // 在首部插入新項 */ }, 1000) }, methods: { shiftArr(){ this.arr.shift(); }, unshiftArr(){ this.arr.unshift({id: -1, data: -1}); } } } </script>
上邊的例子很簡單,就是v-for渲染一個長度為10000的列表,然后在Vue mounted 1s后,執(zhí)行一個刪除列表首項或在列表頭插入新項,觀察兩種key綁定的具體頁面更新開銷。
頁面開銷使用chrome的performance選項卡來測算
刪除列表首項
列表頭 unshift 新元素
出現(xiàn)錯誤
某些節(jié)點在錯誤的位置被復用。(例如當列表項中使用到復選框時)
<!-- 測試代碼 --> <template> <div> <button @click="test">刪除列表第一項</button> <div v-for="(item, index) in arr" :key="index 或 item.id"> <input type="checkbox" /> {{item.data}} </div> </div> </template> <script> export default { name: 'HelloWorld', data(){ return { arr: Array.from({length: 5}, (v, i) => {return {id: i, data: i}}) } }, methods: { test(){ this.arr.shift(); } } } </script>
總結
到此這篇關于Vue頁面渲染中key的應用的文章就介紹到這了,更多相關Vue頁面渲染key的應用內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
vue3中watch和watchEffect實戰(zhàn)梳理
這篇文章主要介紹了vue3中watch和watchEffect實戰(zhàn)梳理,watch和watchEffect都是vue3中的監(jiān)聽器,但是在寫法和使用上是有區(qū)別的。下文介紹他們之間的方法及區(qū)別,需要的朋友可以參考一下2022-07-07vue?button的@click方法無效鉤子函數(shù)沒有執(zhí)行問題
這篇文章主要介紹了vue?button的@click方法無效鉤子函數(shù)沒有執(zhí)行問題及解決,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-03-03Vue實現(xiàn)雙向綁定的原理以及響應式數(shù)據(jù)的方法
這篇文章主要介紹了Vue實現(xiàn)雙向綁定的原理以及響應式數(shù)據(jù)的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-07-07解決el-upload批量上傳只執(zhí)行一次成功回調(diào)on-success的問題
這篇文章主要介紹了解決el-upload批量上傳只執(zhí)行一次成功回調(diào)on-success的問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-03-03vue router解決路由帶參數(shù)跳轉(zhuǎn)時出現(xiàn)404問題
我的頁面是從一個vue頁面router跳轉(zhuǎn)到另一個vue頁面,并且利用windows.open() 瀏覽器重新創(chuàng)建一個頁簽,但是不知道為什么有時候可以有時候又不行,所以本文給大家介紹了vue router解決路由帶參數(shù)跳轉(zhuǎn)時出現(xiàn)404問題,需要的朋友可以參考下2024-03-03