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

一文詳解Vue3中簡(jiǎn)單diff算法的實(shí)現(xiàn)

 更新時(shí)間:2022年09月04日 16:35:50   作者:無(wú)敵小書包  
這篇文章主要為大家詳細(xì)介紹Vue3中簡(jiǎn)單diff算法的實(shí)現(xiàn)與使用,文中的示例代碼講解詳細(xì),具有一定的借鑒價(jià)值,感興趣的可以了解一下

簡(jiǎn)單Diff算法

核心Diff只關(guān)心新舊虛擬節(jié)點(diǎn)都存在一組子節(jié)點(diǎn)的情況

減少DOM操作

例子

// 舊節(jié)點(diǎn)
const oldVNode = {
  type: 'div',
  children: [
    { type: 'p', children: '1' },
    { type: 'p', children: '2' },
    { type: 'p', children: '3' }
  ]
}
// 新節(jié)點(diǎn)
const newVNode = {
  type: 'div',
  children: [
    { type: 'p', children: '4' },
    { type: 'p', children: '5' },
    { type: 'p', children: '6' }
  ]
}

如果直接去操作DOM,那么上面的更新需要6次DOM操作,卸載所有舊子節(jié)點(diǎn),掛載所有新子節(jié)點(diǎn)。

但是觀察上面新舊vNode的子節(jié)點(diǎn)可以發(fā)現(xiàn):

  • 更新前后所有子節(jié)點(diǎn)都是 p 標(biāo)簽,即標(biāo)簽元素步變
  • 只有p標(biāo)簽的子節(jié)點(diǎn)發(fā)生變化了

所以最理想的更新方式是直接更新這個(gè)p標(biāo)簽的文本節(jié)點(diǎn)的內(nèi)容,這樣只需要一次DOM操作,即可完成一個(gè)p標(biāo)簽的更新。更新完所有節(jié)點(diǎn)只需要3次DOM操作就可以完成全部節(jié)點(diǎn)的更新。

上面的做法可以減少DOM操作次數(shù),但問(wèn)題也很明顯,只有節(jié)點(diǎn)數(shù)量相同這個(gè)做法才能正常工作。但新舊兩組子節(jié)點(diǎn)數(shù)量未必相同。

新的一組子節(jié)點(diǎn)數(shù)量少于舊的一組子節(jié)點(diǎn)的數(shù)量時(shí),意味著有節(jié)點(diǎn)在更新后應(yīng)該被卸載。(圖二)

新的一組子節(jié)點(diǎn)數(shù)量多余舊的一組子節(jié)點(diǎn)的數(shù)量時(shí),意味著有節(jié)點(diǎn)在更新后應(yīng)該被新增并掛載。(圖三)

結(jié)論

通過(guò)上面分析得出,進(jìn)行新舊兩組子節(jié)點(diǎn)的更新時(shí),不應(yīng)該總是遍歷舊的一組子節(jié)點(diǎn)或新的一組子節(jié)點(diǎn),而是應(yīng)該遍歷其中較短的一組。這樣才能盡可能多的調(diào)用patch進(jìn)行更新。接著對(duì)比新舊兩組子節(jié)點(diǎn)的長(zhǎng)度,如果新的一組子節(jié)點(diǎn)更長(zhǎng),說(shuō)明有新節(jié)點(diǎn)需要掛載,否則說(shuō)明有舊的子節(jié)點(diǎn)需要卸載。

實(shí)現(xiàn)

function easyDiff (n1, n2, container) {
  // 取出新舊子節(jié)點(diǎn)列表
  const oldChildren = n1.children
  const newChildren = n2.children
  // 獲取新舊子節(jié)點(diǎn)列表的長(zhǎng)度
  const oldLen = oldChildren.length
  const newLen = newChildren.length
  // 取得較小的一個(gè)(可以理解為兩組子節(jié)點(diǎn)的公共長(zhǎng)度)
  const commonLength = Math.min(oldLen, newLen)
  // 遍歷 commonLength 次
  for (let i = 0; i < commonLength; i++) {
    patch(oldChildren[i], newChildren[i], container)
  }
  // 如果 newLen > oldLen,說(shuō)明有新子節(jié)點(diǎn)需要掛載
  if (newLen > oldLen) {
    for (let i = commonLength; i < newLen; i++) {
      patch(null, newChildren[i], container)
    }
  }
  // 如果 oldLen > newLen,說(shuō)明有舊節(jié)點(diǎn)需要卸載
  if (oldLen > newLen) {
    for (let i = commonLength; i < oldLen; i++) {
      unmount(oldChildren[i])
    }
  }
}

DOM復(fù)用與key的作用

例子

上面通過(guò)減少DOM操作次數(shù)提升了更新性能,但還存在可優(yōu)化空間

const KEY = {
  oldVNode: [
    { type: 'p' },
    { type: 'div' },
    { type: 'span' }
  ],
  newVNode: [
    { type: 'span' },
    { type: 'p' },
    { type: 'div' }
  ]
}

針對(duì)這個(gè)例子,如果還使用上面的算法,則需要6次DOM操作。

調(diào)用 patch 在 p標(biāo)簽和span標(biāo)簽之間打補(bǔ)丁,由于不是相同標(biāo)簽,所以p標(biāo)簽被卸載,然后掛載span標(biāo)簽,需要兩步操作,div - p,span - div同理。

很容易發(fā)現(xiàn)新舊兩組子節(jié)點(diǎn)只是順序不同。所以最優(yōu)的處理方式是,通過(guò)DOM的移動(dòng)來(lái)完成子節(jié)點(diǎn)的更新,這比不斷執(zhí)行卸載和掛載性能好得多。但是要通過(guò)移動(dòng)DOM來(lái)完成更新,必須要保證新舊兩組子節(jié)點(diǎn)的確存在可復(fù)用的節(jié)點(diǎn)。(如果新的子節(jié)點(diǎn)沒(méi)有在舊的子節(jié)點(diǎn)中出現(xiàn),則無(wú)法通過(guò)移動(dòng)節(jié)點(diǎn)的方式完成更新操作。)

用上面的例子來(lái)說(shuō),怎么確定新的一組節(jié)點(diǎn)中的第三個(gè)節(jié)點(diǎn) { type: 'div' } 與舊的一組子節(jié)點(diǎn)中的第二個(gè)節(jié)點(diǎn)相同呢?可以通過(guò)vNode.type判斷,但這種方式并不可靠。

  oldChildren: [
    { type: 'p', children: '1' },
    { type: 'p', children: '2' },
    { type: 'p', children: '3' }
  ],
  newChildren: [
    { type: 'p', children: '3' },
    { type: 'p', children: '1' },
    { type: 'p', children: '2' }
  ]

觀察上面節(jié)點(diǎn),可以發(fā)現(xiàn),這個(gè)案例可以通過(guò)移動(dòng)DOM的方式來(lái)完成更新,但是vNode.type的值都相同,導(dǎo)致無(wú)法確定新舊節(jié)點(diǎn)中的對(duì)應(yīng)關(guān)系,就不能確定怎么移動(dòng)DOM完成更新。

虛擬節(jié)點(diǎn)的key

因此,需要引入額外的 key 作為vNode的標(biāo)識(shí)。

const KEY = {
  oldChildren: [
    { type: 'p', children: '1', key: '1' },
    { type: 'p', children: '2', key: '2' },
    { type: 'p', children: '3', key: '3' }
  ],
  newChildren: [
    { type: 'p', children: '3', key: '3' },
    { type: 'p', children: '1', key: '1' },
    { type: 'p', children: '2', key: '2' }
  ]
}

key 屬性就像虛擬DOM的 身份證號(hào),只要兩個(gè)虛擬節(jié)點(diǎn)的type和key屬性都相同,那么就可以認(rèn)為它們是相同的;即可以進(jìn)行DOM的復(fù)用。

但是DOM可復(fù)用并不意味著不需要更新

oldVNode: { type: 'p', children: 'text - 1', key: '1' }
newVNode: { type: 'p', children: 'text - 2', key: '1' }

兩個(gè)節(jié)點(diǎn)有相同的key可type,但它們的文本內(nèi)容不同,還是需要通過(guò)patch進(jìn)行打補(bǔ)丁操作。

實(shí)現(xiàn)

function easyDiffV2 (n1, n2, container) {
  // 取出新舊子節(jié)點(diǎn)列表
  const oldChildren = n1.children
  const newChildren = n2.children
  // 遍歷新的children
  for (let i = 0; i < newChildren.length; i++) {
    const newVNode = newChildren[i]
    for (let j = 0; j < oldChildren.length; j++) {
      const oldVNode = oldChildren[i]
      // 如果找到可復(fù)用的兩個(gè)節(jié)點(diǎn)
      if (newVNode.key === oldVNode.key) {
        // 對(duì)可復(fù)用的兩個(gè)節(jié)點(diǎn)打補(bǔ)丁
        patch(oldVNode, newVNode, container)
        // 一個(gè)新節(jié)點(diǎn)處理完后開始下一個(gè)新節(jié)點(diǎn)
        break
      }
    }
  }
}

外層循環(huán)遍歷新的一組子節(jié)點(diǎn),內(nèi)層循環(huán)遍歷舊的一組子節(jié)點(diǎn)。內(nèi)層循環(huán)中對(duì)比新舊子節(jié)點(diǎn)的key值,在舊的子節(jié)點(diǎn)中找到可以復(fù)用的節(jié)點(diǎn);一旦找到則調(diào)用 patch 打補(bǔ)丁。

找到需要移動(dòng)的元素

現(xiàn)在已經(jīng)可以通過(guò)key找到可復(fù)用的節(jié)點(diǎn)了,接下來(lái)要做的是判斷一個(gè)節(jié)點(diǎn)是否需要移動(dòng)

探索節(jié)點(diǎn)順序關(guān)系

節(jié)點(diǎn)順序不變 - 查找過(guò)程:

第一步:取新的一組子節(jié)點(diǎn)中的第一個(gè)節(jié)點(diǎn) p - 1,它的key為1,在舊的一組子節(jié)點(diǎn)中找到具有相同key值的可復(fù)用節(jié)點(diǎn),能夠找到,并且該節(jié)點(diǎn)在舊的一組子節(jié)點(diǎn)中索引為0;p - 2、p = 3同理。

  • key 為 1 的節(jié)點(diǎn)在 舊節(jié)點(diǎn)列表中的索引為0
  • key 為 2 的節(jié)點(diǎn)在 舊節(jié)點(diǎn)列表中的索引為1
  • key 為 3 的節(jié)點(diǎn)在 舊節(jié)點(diǎn)列表中的索引為2

每一次查找可復(fù)用節(jié)點(diǎn)都會(huì)記錄該可復(fù)用節(jié)點(diǎn)在舊的一組子節(jié)點(diǎn)中的位置索引,如果按照先后順序排列,則可以得到一個(gè)序列:0、1、2,是一個(gè)遞增序列。

節(jié)點(diǎn)順序變化 - 查找過(guò)程

第一步:取新的一組子節(jié)點(diǎn)中的第一個(gè)節(jié)點(diǎn) p - 3,它的key為3,在舊的一組子節(jié)點(diǎn)中找到具有相同key值的可復(fù)用節(jié)點(diǎn),能夠找到,并且該節(jié)點(diǎn)在舊的一組子節(jié)點(diǎn)中索引為2;

第二步:取新的一組子節(jié)點(diǎn)中的第一個(gè)節(jié)點(diǎn) p - 1,它的key為1,在舊的一組子節(jié)點(diǎn)中找到具有相同key值的可復(fù)用節(jié)點(diǎn),能夠找到,并且該節(jié)點(diǎn)在舊的一組子節(jié)點(diǎn)中索引為0;

到了這一步發(fā)現(xiàn)遞增的順序被打破了。節(jié)點(diǎn) p - 1 在舊的一組children 的索引為0,它小于 p - 3 在舊children中的索引2.這說(shuō)明節(jié)點(diǎn) p - 1 在舊children中排在 p - 3前面,但在新的children中,它排在節(jié)點(diǎn) p - 3后面。因此得出:節(jié)點(diǎn)p - 1對(duì)應(yīng)的真實(shí)DOM需要移動(dòng)

第三步:取新的一組子節(jié)點(diǎn)中的第一個(gè)節(jié)點(diǎn) p - 2,它的key為2,在舊的一組子節(jié)點(diǎn)中找到具有相同key值的可復(fù)用節(jié)點(diǎn),能夠找到,并且該節(jié)點(diǎn)在舊的一組子節(jié)點(diǎn)中索引為1;

節(jié)點(diǎn) p - 2 在舊的一組children 的索引為0,它小于 p - 3 在舊children中的索引2.這說(shuō)明節(jié)點(diǎn) p - 2 在舊children中排在 p - 3前面,但在新的children中,它排在節(jié)點(diǎn) p - 3后面。因此得出:**節(jié)點(diǎn)p - 2對(duì)應(yīng)的真實(shí)DOM需要移動(dòng)

可以將節(jié)點(diǎn) p - 3 在舊children中的索引定義為:在舊children中尋找具有相同key值節(jié)點(diǎn)的過(guò)程中,遇到的最大索引值

如果后續(xù)尋找過(guò)程中,存在比當(dāng)前遇到的最大索引值還要小的節(jié)點(diǎn),則意味著該節(jié)點(diǎn)需要移動(dòng)。

實(shí)現(xiàn)

function easyBigIndex (n1, n2, container) {
  // 取出新舊子節(jié)點(diǎn)列表
  const oldChildren = n1.children
  const newChildren = n2.children
  // 用來(lái)存儲(chǔ)尋找過(guò)程中遇到的最大索引值
  let lastIndex = 0
  for (let i = 0; i < newChildren.length; i++) {
    const newVNode = newChildren[i]
    for (let j = 0; j < oldChildren; j++) {
      const oldVNode = oldChildren[j]
      if (newVNode.key === oldVNode.key) {
        patch(oldVNode, newVNode, container)
        if (j < lastIndex) {
          // 需要移動(dòng)
        } else {
          // 更新lastIndex的值(lastIndex要保持當(dāng)前已查找的索引中的最大值)
          lastIndex = j
        }
        break
      }
    }
  }
}

如何移動(dòng)元素

移動(dòng)節(jié)點(diǎn)指的是,移動(dòng)一個(gè)虛擬節(jié)點(diǎn)所對(duì)應(yīng)的真實(shí)DOM節(jié)點(diǎn),并不是移動(dòng)虛擬節(jié)點(diǎn)本身。既然移動(dòng)的是真實(shí)DOM節(jié)點(diǎn),就需要取得它的引用,其對(duì)應(yīng)的真實(shí)DOM節(jié)點(diǎn)會(huì)存儲(chǔ)到它的vNode.el屬性中

例子

引用上面的案例:

取新的一組子節(jié)點(diǎn)中的第一個(gè)節(jié)點(diǎn) p - 3,它的key 為3,在舊的虛擬節(jié)點(diǎn)列表中找到具有相同 key 值的可復(fù)用節(jié)點(diǎn)。發(fā)現(xiàn)能夠找到,并且該節(jié)點(diǎn)在舊的一組子節(jié)點(diǎn)中的素引為2。此時(shí)變量 lastIndex 的值為 0,索引2 不小于0,所以節(jié)點(diǎn) p - 3對(duì)應(yīng)的真實(shí)DOM 不需要移動(dòng),但需要更新變量 lastIndex 的值為 2。

第二步:取新的一組子節(jié)點(diǎn)中第二個(gè)節(jié)點(diǎn) p - 1,它的key 為1,在舊的一組子節(jié)點(diǎn)中找到具有相同 key 值的可復(fù)用節(jié)點(diǎn)。發(fā)現(xiàn)能夠找到,并且該節(jié)點(diǎn)在日的一組子節(jié)點(diǎn)中的索引為0。此時(shí)變量 lastIndex 的值為 2,索引0小于 2,所以節(jié)點(diǎn)p-1對(duì)應(yīng)的真實(shí) DOM需要移動(dòng)

到了這一步,我們發(fā)現(xiàn),節(jié)點(diǎn)p - 1對(duì)應(yīng)的真實(shí) DOM 需要移動(dòng),但應(yīng)該移動(dòng)到哪里呢?新children 的順序其實(shí)就是更新后真實(shí) DOM 節(jié)點(diǎn)應(yīng)有的順序。所以節(jié)點(diǎn) p-1在新 children 中的位置就代表了真實(shí) DOM 更新后的位置。由于節(jié)點(diǎn) p - 1在新 children 中排在節(jié)點(diǎn)p - 3后面,所以我們應(yīng)該把節(jié)點(diǎn)p - 1所對(duì)應(yīng)的真實(shí) DOM移動(dòng)到節(jié)點(diǎn)p - 3所對(duì)應(yīng)的真實(shí) DOM 后面。這樣操作之后,此時(shí)真實(shí) DOM 的順序?yàn)?p-2、p-3、p-1。

第三步:取新的一組子節(jié)點(diǎn)中第三個(gè)節(jié)點(diǎn) p-2,它的key 為2。嘗試在舊的一組子節(jié)點(diǎn)中找到具有相同 key 值的可復(fù)用節(jié)點(diǎn)。發(fā)現(xiàn)能夠找到,并且該節(jié)點(diǎn)在舊的一組子節(jié)點(diǎn)中的素引為1。此時(shí)變量 lastIndex 的值為 2,索引1小于2,所以節(jié)點(diǎn)p-2對(duì)應(yīng)的真實(shí) DOM需要移動(dòng)

第二步操作完成后 新 / 舊 / 虛擬 節(jié)點(diǎn)之間的對(duì)應(yīng)關(guān)系

實(shí)現(xiàn)

function easyMove (n1, n2, container) {
  // 取出新舊子節(jié)點(diǎn)列表
  const oldChildren = n1.children
  const newChildren = n2.children
  // 用來(lái)存儲(chǔ)尋找過(guò)程中遇到的最大索引值
  let lastIndex = 0
  for (let i = 0; i < newChildren.length; i++) {
    const newVNode = newChildren[i]
    for (let j = 0; j < oldChildren; j++) {
      const oldVNode = oldChildren[j]
      if (newVNode.key === oldVNode.key) {
        patch(oldVNode, newVNode, container)
        if (j < lastIndex) {
          // 需要移動(dòng)
          // 獲取當(dāng)前vNode的前一個(gè)vNode
          const prevVNode = newChildren[i - 1]
          // 如果 prevVNode 不存在,說(shuō)明當(dāng)前vNode是第一個(gè)節(jié)點(diǎn),它不需要移動(dòng)
          if (prevVNode) {
            // 由于要將newVNode對(duì)用的真實(shí)DOM移動(dòng)到prevVNode對(duì)應(yīng)的真實(shí)DOM后面,
            // 所以需要獲取prevVNode對(duì)應(yīng)的真實(shí)節(jié)點(diǎn)的下一個(gè)兄弟節(jié)點(diǎn),并將其作為錨點(diǎn)
            const anchor = prevVNode.el.nextSibling
            // 調(diào)用insert將newVNode對(duì)應(yīng)真實(shí)DOM插入到錨點(diǎn)元素前面
            // insert 是通過(guò) el.insertBefore 插入元素的
            insert(newVNode.el, container, anchor)
          }
        } else {
          // 更新lastIndex的值(lastIndex要保持當(dāng)前已查找的索引中的最大值)
          lastIndex = j
        }
        break
      }
    }
  }
}

添加新元素

例子

在新的一組子節(jié)點(diǎn)中,多出來(lái)一個(gè) p - 4,它的key值為4,該節(jié)點(diǎn)在舊的一組字節(jié)點(diǎn)中不存在,因此應(yīng)該將其視為新增節(jié)點(diǎn)。對(duì)于新增節(jié)點(diǎn),更新時(shí)應(yīng)該正確地將其掛載:

  • 找到新增節(jié)點(diǎn)
  • 將新增節(jié)點(diǎn)掛載到正確位置

第一步:取新的一組子節(jié)點(diǎn)中第一個(gè)節(jié)點(diǎn)p - 3,它的key值為3,在舊的一組子節(jié)及中找到可復(fù)用的節(jié)點(diǎn)。發(fā)現(xiàn)能找到,并且該節(jié)點(diǎn)在舊的一組子節(jié)點(diǎn)中的索引值為2。此時(shí),變量lastIndex的值為0,所以節(jié)點(diǎn) p - 3 對(duì)應(yīng)的真實(shí) DOM 不需要移動(dòng),但是需要將變量 lastIndex 的值更新為 2。

第二步:取新的一組子節(jié)點(diǎn)中第一個(gè)節(jié)點(diǎn)p - 1,它的key值為1,在舊的一組子節(jié)及中找到可復(fù)用的節(jié)點(diǎn)。發(fā)現(xiàn)能找到,并且該節(jié)點(diǎn)在舊的一組子節(jié)點(diǎn)中的索引值為1。此時(shí)變量lastIndex的值為2,所以節(jié)點(diǎn) p - 1對(duì)應(yīng)的真實(shí)DOM需要移動(dòng),并且應(yīng)該移動(dòng)到節(jié)點(diǎn) p - 3對(duì)應(yīng)的真實(shí)DOM后面。

第三步:取新的一組子節(jié)點(diǎn)中第一個(gè)節(jié)點(diǎn)p - 4,它的key值為4,在舊的一組子節(jié)及中找到可復(fù)用的節(jié)點(diǎn)。沒(méi)有key值為4的節(jié)點(diǎn),因此渲染器會(huì)把節(jié)點(diǎn) p - 4 看作新增節(jié)點(diǎn)并掛載它。應(yīng)該掛載到什么地方呢?觀察p - 4在新的一組子節(jié)點(diǎn)中的位置。由于 p - 4出現(xiàn)在節(jié)點(diǎn) p - 1后面,所以應(yīng)該把 p - 4 掛載到節(jié)點(diǎn) p - 1 對(duì)應(yīng)的真實(shí)DOM后面。

第四步:取新的一組子節(jié)點(diǎn)中第一個(gè)節(jié)點(diǎn)p - 2,它的key值為2,在舊的一組子節(jié)及中找到可復(fù)用的節(jié)點(diǎn)。發(fā)現(xiàn)能找到,并且該節(jié)點(diǎn)在舊的一組子節(jié)點(diǎn)中的索引值為1。此時(shí),變量lastIndex的值為2,索引值1小于lastIndex的值2,所以節(jié)點(diǎn) p - 2對(duì)應(yīng)的真實(shí)DOM需要移動(dòng),并且應(yīng)該移動(dòng)到節(jié)點(diǎn) p - 4對(duì)應(yīng)的真實(shí)DOM后面。

第二步操作完成后的節(jié)點(diǎn)對(duì)應(yīng)關(guān)系

第三步操作完成后的節(jié)點(diǎn)對(duì)應(yīng)關(guān)系

實(shí)現(xiàn)

function easyMount (n1, n2, container) {
  // 取出新舊子節(jié)點(diǎn)列表
  const oldChildren = n1.children
  const newChildren = n2.children
  // 用來(lái)存儲(chǔ)尋找過(guò)程中遇到的最大索引值
  let lastIndex = 0
  for (let i = 0; i < newChildren.length; i++) {
    const newVNode = newChildren[i]

    // 定義變量 find,代表是否在舊的一組子節(jié)點(diǎn)中找到可復(fù)用的節(jié)點(diǎn),初始值為false - 沒(méi)找到
    let find = false
    for (let j = 0; j < oldChildren; j++) {
      const oldVNode = oldChildren[j]
      if (newVNode.key === oldVNode.key) {
        // 一旦找到可復(fù)用的節(jié)點(diǎn),將變量find設(shè)置為true
        find = true
        patch(oldVNode, newVNode, container)
        if (j < lastIndex) {
          // 需要移動(dòng)
          // 獲取當(dāng)前vNode的前一個(gè)vNode
          const prevVNode = newChildren[i - 1]
          // 如果 prevVNode 不存在,說(shuō)明當(dāng)前vNode是第一個(gè)節(jié)點(diǎn),它不需要移動(dòng)
          if (prevVNode) {
            // 由于要將newVNode對(duì)用的真實(shí)DOM移動(dòng)到prevVNode對(duì)應(yīng)的真實(shí)DOM后面,
            // 所以需要獲取prevVNode對(duì)應(yīng)的真實(shí)節(jié)點(diǎn)的下一個(gè)兄弟節(jié)點(diǎn),并將其作為錨點(diǎn)
            const anchor = prevVNode.el.nextSibling
            // 調(diào)用insert將newVNode對(duì)應(yīng)真實(shí)DOM插入到錨點(diǎn)元素前面
            // insert 是通過(guò) el.insertBefore 插入元素的
            insert(newVNode.el, container, anchor)
          }
        } else {
          // 更新lastIndex的值(lastIndex要保持當(dāng)前已查找的索引中的最大值)
          lastIndex = j
        }
        break
      }
    }
    // 這里find如果還是false,說(shuō)明當(dāng)前newVNode沒(méi)有在舊的一組子節(jié)點(diǎn)中找到可復(fù)用的節(jié)點(diǎn)
    // 也就是說(shuō)當(dāng)前 newVNode 是新增節(jié)點(diǎn),需要掛載
    if (!find) {
      // 為了將節(jié)點(diǎn)掛載到正確位置,需要先獲取錨點(diǎn)元素
      // 首先獲取當(dāng)前newVNode的前一個(gè)vNode節(jié)點(diǎn)
      const prevVNode = newChildren[i - 1]
      let anchor = null
      if (prevVNode) {
        // 如果有前一個(gè)vNode節(jié)點(diǎn),則使用它的下一個(gè)兄弟節(jié)點(diǎn)作為錨點(diǎn)元素
        anchor = prevVNode.el.nextSibling
      } else {
        // 如果沒(méi)有前一個(gè)vNode節(jié)點(diǎn),說(shuō)明即將掛載的新節(jié)點(diǎn)是第一個(gè)子節(jié)點(diǎn)
        // 這是使用容器元素的firstChild作為錨點(diǎn)
        anchor = container.firstChild
      }
      // 掛載 newVNode
      patch(null, newVNode, container, anchor)
    }
  }
}

移除不存在的元素

例子

在新的一組節(jié)點(diǎn)中,節(jié)點(diǎn) p - 2 不存在了,說(shuō)明該節(jié)點(diǎn)被刪除,渲染器應(yīng)該能找到那些需要?jiǎng)h除的節(jié)點(diǎn)并正確地將其刪除。

找到需要?jiǎng)h除的節(jié)點(diǎn) - 步驟:

  • 第一步:取新的一組子節(jié)點(diǎn)中的第一個(gè)節(jié)點(diǎn)p - 3,它的key 值為3,在舊的一組子節(jié)點(diǎn)中尋找可復(fù)用的節(jié)點(diǎn)。發(fā)現(xiàn)能夠找到,并且該節(jié)點(diǎn)在舊的一組子節(jié)點(diǎn)中的索引值為2。此時(shí)變量 lastIndex 的值為0,索引2不小于lastIndex 的值0,所以節(jié)點(diǎn)p - 3對(duì)應(yīng)的真實(shí) DOM 不需要移動(dòng),但需要更新變量 lastIndex 的值為 2。
  • 第二步:取新的一組子節(jié)點(diǎn)中的第二個(gè)節(jié)點(diǎn) p - 1,它的key 值為1。嘗試在舊的一組子節(jié)點(diǎn)中尋找可復(fù)用的節(jié)點(diǎn)。發(fā)現(xiàn)能夠找到,并且該節(jié)點(diǎn)在舊的一組子節(jié)點(diǎn)中的索引值為0。此時(shí)變量 lastIndex 的值為2,索引0小于 lastIndex 的值 2, 所以節(jié)點(diǎn)p-1對(duì)應(yīng)的真實(shí) DOM需要移動(dòng),并且應(yīng)該移動(dòng)到節(jié)點(diǎn)p-3對(duì)應(yīng)的真實(shí) DOM 后面經(jīng)過(guò)這一步的移動(dòng)操作后,發(fā)現(xiàn) p - 3和p - 1都有了對(duì)應(yīng)的真實(shí)DOM節(jié)點(diǎn)。
  • 至此更新結(jié)束,但 p - 2 對(duì)應(yīng)的真實(shí)DOM仍然存在,所以需要增加額外的邏輯來(lái)刪除遺留節(jié)點(diǎn)。當(dāng)基本的更新結(jié)束時(shí),需要遍歷舊的一組子節(jié)點(diǎn),然后去新的一組子節(jié)點(diǎn)中尋找具有相同key值的節(jié)點(diǎn)。如果找不到,說(shuō)明應(yīng)該刪除該節(jié)點(diǎn)。

p - 2與任何newVNode沒(méi)有對(duì)應(yīng)關(guān)系

實(shí)現(xiàn)

function easyUnmount (n1, n2, container) {
  // 取出新舊子節(jié)點(diǎn)列表
  const oldChildren = n1.children
  const newChildren = n2.children
  // 用來(lái)存儲(chǔ)尋找過(guò)程中遇到的最大索引值
  let lastIndex = 0
  for (let i = 0; i < newChildren.length; i++) {
    const newVNode = newChildren[i]

    // 定義變量 find,代表是否在舊的一組子節(jié)點(diǎn)中找到可復(fù)用的節(jié)點(diǎn),初始值為false - 沒(méi)找到
    let find = false
    for (let j = 0; j < oldChildren; j++) {
      const oldVNode = oldChildren[j]
      if (newVNode.key === oldVNode.key) {
        // 一旦找到可復(fù)用的節(jié)點(diǎn),將變量find設(shè)置為true
        find = true
        patch(oldVNode, newVNode, container)
        if (j < lastIndex) {
          // 需要移動(dòng)
          // 獲取當(dāng)前vNode的前一個(gè)vNode
          const prevVNode = newChildren[i - 1]
          // 如果 prevVNode 不存在,說(shuō)明當(dāng)前vNode是第一個(gè)節(jié)點(diǎn),它不需要移動(dòng)
          if (prevVNode) {
            // 由于要將newVNode對(duì)用的真實(shí)DOM移動(dòng)到prevVNode對(duì)應(yīng)的真實(shí)DOM后面,
            // 所以需要獲取prevVNode對(duì)應(yīng)的真實(shí)節(jié)點(diǎn)的下一個(gè)兄弟節(jié)點(diǎn),并將其作為錨點(diǎn)
            const anchor = prevVNode.el.nextSibling
            // 調(diào)用insert將newVNode對(duì)應(yīng)真實(shí)DOM插入到錨點(diǎn)元素前面
            // insert 是通過(guò) el.insertBefore 插入元素的
            insert(newVNode.el, container, anchor)
          }
        } else {
          // 更新lastIndex的值(lastIndex要保持當(dāng)前已查找的索引中的最大值)
          lastIndex = j
        }
        break
      }
    }
    // 這里find如果還是false,說(shuō)明當(dāng)前newVNode沒(méi)有在舊的一組子節(jié)點(diǎn)中找到可復(fù)用的節(jié)點(diǎn)
    // 也就是說(shuō)當(dāng)前 newVNode 是新增節(jié)點(diǎn),需要掛載
    if (!find) {
      // 為了將節(jié)點(diǎn)掛載到正確位置,需要先獲取錨點(diǎn)元素
      // 首先獲取當(dāng)前newVNode的前一個(gè)vNode節(jié)點(diǎn)
      const prevVNode = newChildren[i - 1]
      let anchor = null
      if (prevVNode) {
        // 如果有前一個(gè)vNode節(jié)點(diǎn),則使用它的下一個(gè)兄弟節(jié)點(diǎn)作為錨點(diǎn)元素
        anchor = prevVNode.el.nextSibling
      } else {
        // 如果沒(méi)有前一個(gè)vNode節(jié)點(diǎn),說(shuō)明即將掛載的新節(jié)點(diǎn)是第一個(gè)子節(jié)點(diǎn)
        // 這是使用容器元素的firstChild作為錨點(diǎn)
        anchor = container.firstChild
      }
      // 掛載 newVNode
      patch(null, newVNode, anchor)
    }
  }

  // 更新操作完成后,遍歷舊的一組子節(jié)點(diǎn)
  for (let i = 0; i < oldChildren.length; i++) {
    const oldVNode = oldChildren[i]
    // 拿舊子節(jié)點(diǎn)oldVNode去新的一組子節(jié)點(diǎn)中尋找具有相同key值的節(jié)點(diǎn)
    const has = newChildren.find(
      vNode => vNode.key === oldVNode.key
    )
    if (has) {
      // 如果沒(méi)找到具有相同key值的節(jié)點(diǎn),則說(shuō)明需要?jiǎng)h除該節(jié)點(diǎn),調(diào)用unmount函數(shù)將其卸載
      unmount(oldVNode)
    }
  }
}

總結(jié)

遍歷新舊子結(jié)點(diǎn)中較少的一組,逐個(gè)調(diào)用patch進(jìn)行打補(bǔ)丁,然后比較新舊兩組子節(jié)點(diǎn)的數(shù)量,如果新的一組子節(jié)點(diǎn)數(shù)量更多,說(shuō)明有新節(jié)點(diǎn)需要掛載;否則說(shuō)明在舊的一組子節(jié)點(diǎn)中,有節(jié)點(diǎn)需要卸載。

引入key屬性,就像虛擬節(jié)點(diǎn)的身份證號(hào),通過(guò)key找到可復(fù)用的節(jié)點(diǎn),然后盡可能通過(guò)DOM移動(dòng)操作來(lái)完成更新,避免過(guò)多地對(duì)DOM元素進(jìn)行銷毀和重建。

簡(jiǎn)單Diff算法地核心邏輯是,拿新的一組子節(jié)點(diǎn)中地節(jié)點(diǎn)去舊的一組子節(jié)點(diǎn)中尋找可復(fù)用地節(jié)點(diǎn),如果找到了,則記錄該節(jié)點(diǎn)地位置索引。在整個(gè)更新過(guò)程中,如果一個(gè)節(jié)點(diǎn)地索引值小于最大索引,則說(shuō)明該節(jié)點(diǎn)對(duì)應(yīng)地真實(shí)DOM元素需要移動(dòng)。

以上就是一文詳解Vue3中簡(jiǎn)單diff算法的實(shí)現(xiàn)的詳細(xì)內(nèi)容,更多關(guān)于Vue簡(jiǎn)單diff算法的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • elementplus?中?DatePicker?日期選擇器樣式修改無(wú)效的問(wèn)題及解決方案

    elementplus?中?DatePicker?日期選擇器樣式修改無(wú)效的問(wèn)題及解決方案

    這篇文章主要介紹了elementplus中DatePicker日期選擇器樣式修改無(wú)效的問(wèn)題,DatePicker日期選擇器彈出面板默認(rèn)掛載在body上,所以在組件中添加了?scoped?屬性的?style?標(biāo)簽下是修改不到其樣式的,講解了datepicker的使用方法,及常見的配置項(xiàng)和對(duì)應(yīng)的值,需要的朋友可以參考下
    2024-01-01
  • 淺入深出Vue之組件使用

    淺入深出Vue之組件使用

    這篇文章主要給大家介紹了關(guān)于淺入深出Vue之組件使用的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用Vue具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2019-07-07
  • 淺談vue項(xiàng)目,訪問(wèn)路徑#號(hào)的問(wèn)題

    淺談vue項(xiàng)目,訪問(wèn)路徑#號(hào)的問(wèn)題

    這篇文章主要介紹了淺談vue項(xiàng)目,訪問(wèn)路徑#號(hào)的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2020-08-08
  • Vuex子模塊調(diào)用子模塊的actions或mutations實(shí)現(xiàn)方式

    Vuex子模塊調(diào)用子模塊的actions或mutations實(shí)現(xiàn)方式

    這篇文章主要介紹了Vuex子模塊調(diào)用子模塊的actions或mutations實(shí)現(xiàn)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-10-10
  • 在Vue中實(shí)現(xiàn)網(wǎng)頁(yè)截圖與截屏功能詳解

    在Vue中實(shí)現(xiàn)網(wǎng)頁(yè)截圖與截屏功能詳解

    在Web開發(fā)中,有時(shí)候需要對(duì)網(wǎng)頁(yè)進(jìn)行截圖或截屏,Vue作為一個(gè)流行的JavaScript框架,提供了一些工具和庫(kù),可以方便地實(shí)現(xiàn)網(wǎng)頁(yè)截圖和截屏功能,本文將介紹如何在Vue中進(jìn)行網(wǎng)頁(yè)截圖和截屏,需要的朋友可以參考下
    2023-06-06
  • vue實(shí)現(xiàn)上傳圖片添加水印(升級(jí)版)

    vue實(shí)現(xiàn)上傳圖片添加水印(升級(jí)版)

    這篇文章主要為大家詳細(xì)介紹了vue實(shí)現(xiàn)上傳圖片添加水印的升級(jí)版,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-09-09
  • vue3組件通信的方式總結(jié)及實(shí)例用法

    vue3組件通信的方式總結(jié)及實(shí)例用法

    在本篇文章里小編給大家整理的是一篇關(guān)于vue3組件通信的方式總結(jié)及實(shí)例用法,對(duì)此有興趣的朋友們可以跟著學(xué)習(xí)下。
    2021-12-12
  • vue使用screenfull插件實(shí)現(xiàn)全屏功能

    vue使用screenfull插件實(shí)現(xiàn)全屏功能

    這篇文章主要為大家詳細(xì)介紹了vue使用screenfull插件實(shí)現(xiàn)全屏功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-09-09
  • vue2.0 與 bootstrap datetimepicker的結(jié)合使用實(shí)例

    vue2.0 與 bootstrap datetimepicker的結(jié)合使用實(shí)例

    本篇文章主要介紹了vue2.0 與 bootstrap datetimepicker的結(jié)合使用實(shí)例,非常具有實(shí)用價(jià)值,需要的朋友可以參考下
    2017-05-05
  • Vue金融數(shù)字格式化(并保留小數(shù))數(shù)字滾動(dòng)效果實(shí)現(xiàn)

    Vue金融數(shù)字格式化(并保留小數(shù))數(shù)字滾動(dòng)效果實(shí)現(xiàn)

    這篇文章主要介紹了Vue金融數(shù)字格式化(并保留小數(shù)) 數(shù)字滾動(dòng)效果,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2023-04-04

最新評(píng)論