欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Vue中key為index和id的區(qū)別示例詳解

 更新時間:2023年06月13日 10:01:48   作者:Lakeiedward  
這篇文章主要介紹了Vue中key為index和id的區(qū)別詳解,本文通過實例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下

一、Diff算法

在了解key的作用之前,先簡單認(rèn)識一下diff算法??

diff算法的特點是平級比較,采用雙指針和遞歸的方式進(jìn)行逐級比較。

Vue會有一個根節(jié)點,先判斷根節(jié)點是否是文本節(jié)點,如果不是文本節(jié)點,則會判斷是否都有兒子節(jié)點,如果都有并且新舊兒子節(jié)點不相等,此時就會比較這兩個新舊兒子節(jié)點(updateChildren),在做比較的時候會有以下幾種情況:

  • 頭頭對比
  • 尾尾對比
  • 頭尾對比
  • 尾頭對比
  • 亂序?qū)Ρ?,根?jù)舊節(jié)點會生成一個映射表(也就是map對象),用新的節(jié)點一個個在映射表里找,沒有的話插入,有的話移動復(fù)用,多余的刪掉。

二、Key的作用

在比較兩個節(jié)點的時候sameVnode(oldStartVnode, newStartVnode),主要根據(jù)key進(jìn)行判斷兩個元素是否是一個元素,key不相同的話則說明不是同一個元素。使用key的時候盡量保持key的唯一性(這樣可以優(yōu)化diff算法)

動態(tài)列表添加的key的時候,要避免使用索引(index)!

接下來,我們使用數(shù)組渲染一組兒子節(jié)點小li,并且通過事件在數(shù)組的頭部增加(unshift)一個數(shù)據(jù);當(dāng)key為index的時候,我們查看下圖圖片渲染的情況發(fā)現(xiàn)所有的小li都變化了,而key為id的時候,則只在li的最前面新加了一個小li,這就是diff算法根據(jù)key判斷產(chǎn)生的差異性,具體在下面來看一看。

三、Key為Index

1) 圖解

如下圖,首先上面是舊節(jié)點,下面是新節(jié)點,新節(jié)點上在數(shù)組最前面新加了一個C節(jié)點,因為key是index,所以此時C的key還是0,但是文本是C,并不是A。

因為第一個新舊節(jié)點的key相同,所以此時會先進(jìn)入到頭頭對比中,而不會進(jìn)入到尾尾對比,在對比的過程中,會再次進(jìn)入到patchVnode方法中判斷新舊節(jié)點的文本是否一致,如果一致則直接復(fù)用,不一致則會對dom進(jìn)行操作,將舊節(jié)點文本替換成新節(jié)點文本node.textContent = text

第一組對比完成之后,新舊節(jié)點的索引會依次增加,對比第二組,第二組的key也是一樣的,會重復(fù)第一組的對比方式,最后將舊節(jié)點文本替換成新節(jié)點文本node.textContent = text

此時因為舊節(jié)點的開始索引和結(jié)束索引相等,則會退出while循環(huán),根據(jù)判斷新舊節(jié)點的開始和結(jié)束索引得出,最后一個剩余的新節(jié)點會插入(addVnodes)到A元素后面去。

此時更新就結(jié)束了,會發(fā)現(xiàn)進(jìn)行了三次dom操作,雖然新舊節(jié)點除了新增的C節(jié)點,其他都是相同的,但是都沒有復(fù)用原來的節(jié)點,而是直接使用textContent改變文本,所以index作為key不中!

2) 完整的步驟

看下一個完整的步驟:

  • 如果key是index,在頭部添加一個節(jié)點,新加的節(jié)點key還是0,和第一個舊節(jié)點是一樣的key(但是文本不一樣),sameVnode就會判斷他們倆是一樣的節(jié)點,就會頭頭對比(而不是尾尾對比),此時雖然key相同的了,但是會遞歸進(jìn)入到patchNode中時,會判斷文本是否相同(key為index時,文本不相同),如果不相同,則會進(jìn)行dom文本替換,把舊的文本替換成新的文本,就會出現(xiàn)上圖所有的小li進(jìn)行更新。
  • 以上步驟會一直重復(fù)頭頭對比,雖然每次對比時,key都是一樣的,但是文本內(nèi)容不一樣,則會一直觸發(fā)dom更新操作,也就是類似lis[0].textContent = 'C',一直到循環(huán)結(jié)束oldStartIdx <= oldEndIdx && newStartIdx <= newEndIdx,此時把多的新節(jié)點添加(addVnodes)進(jìn)去,或者多的老節(jié)點刪除(removeVnodes)掉。

四、Key為Id

1) 圖解

如下圖,新加的節(jié)點key為c,當(dāng)進(jìn)行sameVnode(oldStartVnode, newStartVnode)對比的時候,發(fā)現(xiàn)key不一樣

則開始尾尾對比sameVnode(oldEndVnode, newEndVnode),此時key是一樣的,則進(jìn)入到patchVnode方法,判斷新舊節(jié)點的文本是否一致,一致的話,就復(fù)用原來的節(jié)點了

對比完第一組,此時新舊節(jié)點的尾索引減1,還是尾尾相等,開始尾對比,重復(fù)上述的步驟,復(fù)用原來的舊節(jié)點,沒有dom操作。

>

對比完第二組,舊節(jié)點的頭索引和尾索引相等,則結(jié)束while循環(huán),最后一個剩余的新節(jié)點會插入(addVnodes)到A元素前面去。

以上的步驟完成之后,只有最后一次執(zhí)行了插入dom操作,優(yōu)化了diff算法和減少了dom操作

2) 完整的步驟

完整的步驟:

  • 如果key是唯一的id,向前追加一個,sameVnode判斷新舊節(jié)點時發(fā)現(xiàn)新舊節(jié)點的key不相同,開始尾對比,尾對比會進(jìn)入到patchVnode方法,當(dāng)為文本節(jié)點時,判斷新舊節(jié)點的文本是否相同,結(jié)果發(fā)現(xiàn)相同,則不做更新dom操作,直接復(fù)用原來的,一直到循環(huán)結(jié)束oldStartIdx <= oldEndIdx && newStartIdx <= newEndIdx,此時只需要把多的新節(jié)點添加(addVnodes)進(jìn)去,或者多的老節(jié)點刪除(removeVnodes)掉即可,沒有多余的dom操作。

五、源碼

粘貼一下部分的Vue源碼

1)sameVnode

只會判斷key、 tag、是否有data的存在、是否是注釋節(jié)點、是否是相同的input type,來判斷是否可以復(fù)用這個節(jié)點。

function sameVnode(a, b) {
  return (
    a.key === b.key &&
    a.asyncFactory === b.asyncFactory &&
    ((a.tag === b.tag &&
      a.isComment === b.isComment &&
      isDef(a.data) === isDef(b.data) &&
      sameInputType(a, b)) ||
      (isTrue(a.isAsyncPlaceholder) && isUndef(b.asyncFactory.error)))
  )
}
function sameInputType(a, b) {
  if (a.tag !== 'input') return true
  let i
  const typeA = isDef((i = a.data)) && isDef((i = i.attrs)) && i.type
  const typeB = isDef((i = b.data)) && isDef((i = i.attrs)) && i.type
  return typeA === typeB || (isTextInputType(typeA) && isTextInputType(typeB))
}

2)patchVnode

如果新 vnode 不是文字 vnode

  • 那么就要開始對子節(jié)點 child 進(jìn)行對比了。

如果新舊 children 都存在(都存在 li 子節(jié)點列表,進(jìn)入 )

  • 那么就是 diff算法 想要考察的最核心的點了,也就是新舊節(jié)點的 diff 過程。

如果有新 children 而沒有舊 children

  • 說明是新增 child,直接 addVnodes 添加新子節(jié)點。

如果有舊 children 而沒有新 children

  • 說明是刪除 child,直接 removeVnodes 刪除舊子節(jié)點

如果新 vnode 是文字 vnode

  • 就直接調(diào)用瀏覽器的 dom api 把節(jié)點的直接替換掉文字內(nèi)容就好。
function patchVnode(
  oldVnode,
  vnode,
  insertedVnodeQueue,
  ownerArray,
  index,
  removeOnly?: any
){
  ...
  // 判斷新節(jié)點是不是text節(jié)點
  if (isUndef(vnode.text)) {
  // 不是text節(jié)點
    if (isDef(oldCh) && isDef(ch)) {
      // 老節(jié)點和新節(jié)點都有child,并且child不相等
      if (oldCh !== ch)
        updateChildren(elm, oldCh, ch, insertedVnodeQueue, removeOnly)
    } else if (isDef(ch)) {
      // 新節(jié)點有child,老節(jié)點沒有,則新增
      if (__DEV__) {
        checkDuplicateKeys(ch)
      }
      if (isDef(oldVnode.text)) nodeOps.setTextContent(elm, '')
      addVnodes(elm, null, ch, 0, ch.length - 1, insertedVnodeQueue)
    } else if (isDef(oldCh)) {
      // 老節(jié)點有child,新節(jié)點沒有,則刪除
      removeVnodes(oldCh, 0, oldCh.length - 1)
    } else if (isDef(oldVnode.text)) {
      nodeOps.setTextContent(elm, '')
    }
  } else if (oldVnode.text !== vnode.text) {
    // 是text節(jié)點并且文本不一樣,就把舊的文本替換成新的文本
    nodeOps.setTextContent(elm, vnode.text)
  }
  ...  
}

Tips: 兒子節(jié)點不是文本時,一方有兒子,一方?jīng)]有兒子(刪除、添加),兩方都有兒子,則進(jìn)入diff算法對比

六、總結(jié)

  • 動態(tài)列表添加的key的時候,要避免使用索引(index)
  • 使用唯一的key可以優(yōu)化diff算法,減少更新dom的操作

相關(guān)文章

  • Vue3中refs使用方法舉例

    Vue3中refs使用方法舉例

    在Vue中Refs是對DOM元素或已安裝到DOM的其他組件實例的引用,下面這篇文章主要給大家介紹了關(guān)于Vue3中refs使用的相關(guān)資料,文中通過代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2024-08-08
  • vue中手機(jī)號,郵箱正則驗證以及60s發(fā)送驗證碼的實例

    vue中手機(jī)號,郵箱正則驗證以及60s發(fā)送驗證碼的實例

    下面小編就為大家分享一篇vue中手機(jī)號,郵箱正則驗證以及60s發(fā)送驗證碼的實例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-03-03
  • vue中echarts引入中國地圖的案例

    vue中echarts引入中國地圖的案例

    這篇文章主要介紹了vue中echarts引入中國地圖的案例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-07-07
  • vue如何統(tǒng)一樣式(reset.css與border.css)

    vue如何統(tǒng)一樣式(reset.css與border.css)

    這篇文章主要介紹了vue如何統(tǒng)一樣式(reset.css與border.css),具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-05-05
  • 詳解Vue3框架的搭建及工程目錄

    詳解Vue3框架的搭建及工程目錄

    文章介紹了如何使用Node.js搭建Vue工程,并對Vue工程目錄進(jìn)行了詳細(xì)解讀,包括各種文件夾和文件的作用,此外,還講解了如何設(shè)置網(wǎng)頁標(biāo)題和全局樣式,以及如何配置路由,感興趣的朋友一起看看吧
    2025-03-03
  • elementui的el-popover修改樣式不生效的解決

    elementui的el-popover修改樣式不生效的解決

    在使用element-ui的時候,有一個常用的組件,那就是el-popover,本文就介紹一下elementui的el-popover修改樣式不生效的解決方法,感興趣的可以了解一下
    2021-06-06
  • Vue3中實現(xiàn)微信掃碼登錄的步驟和代碼示例

    Vue3中實現(xiàn)微信掃碼登錄的步驟和代碼示例

    在 Vue 3 中實現(xiàn)微信掃碼登錄,涉及到前端生成二維碼、監(jiān)聽微信回調(diào)以及與后端的交互,本文給大家介紹了一個詳細(xì)的實現(xiàn)步驟和代碼示例,對大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下
    2024-07-07
  • 在Vue的mounted中仍然加載渲染不出echarts的方法問題

    在Vue的mounted中仍然加載渲染不出echarts的方法問題

    這篇文章主要介紹了在Vue的mounted中仍然加載渲染不出echarts的方法問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-03-03
  • 解決vue+webpack項目接口跨域出現(xiàn)的問題

    解決vue+webpack項目接口跨域出現(xiàn)的問題

    這篇文章主要介紹了解決vue+webpack項目接口跨域出現(xiàn)的問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-08-08
  • Vue+Axios實現(xiàn)文件上傳自定義進(jìn)度條

    Vue+Axios實現(xiàn)文件上傳自定義進(jìn)度條

    這篇文章主要為大家詳細(xì)介紹了Vue+Axios實現(xiàn)文件上傳自定義進(jìn)度條,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-08-08

最新評論