日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 前端技术 > vue >内容正文

vue

【源码系列#06】Vue3 Diff算法

發布時間:2024/1/18 vue 44 coder
生活随笔 收集整理的這篇文章主要介紹了 【源码系列#06】Vue3 Diff算法 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

專欄分享:vue2源碼專欄,vue3源碼專欄,vue router源碼專欄,玩具項目專欄,硬核??推薦??
歡迎各位ITer關注點贊收藏??????

Vue2 Diff算法可以參考此篇文章【Vue2.x源碼系列08】Diff算法原理

前后元素不一致

兩個不同虛擬節點不需要進行比較,直接移除老節點,將新的虛擬節點渲染成真實DOM進行掛載即可

// 判斷兩個虛擬節點是否是相同節點,標簽名相同 && key是一樣的
export function isSameVnode(n1, n2) {
  return n1.type === n2.type && n1.key === n2.key
}

//  核心的patch方法,包括初始化DOM 和 diff算法
const patch = (n1, n2, container) => {
  if (n1 == n2) return

  // 判斷兩個元素是否相同,不相同卸載在添加
  if (n1 && !isSameVnode(n1, n2)) {
    unmount(n1) // 刪除老的
    n1 = null
  }

  const { type, shapeFlag } = n2
  switch (type) {
    case Text:
      processText(n1, n2, container)
      break
    default:
      if (shapeFlag & ShapeFlags.ELEMENT) {
        processElement(n1, n2, container)
      }
  }
}

前后元素一致

兩個相同的虛擬節點,先復用節點,再比較兩個節點的屬性和孩子節點

判斷是否是相同的虛擬節點:type類型相同 && key相同

  1. 處理文本類型的虛擬節點
// 處理文本,初始化文本和patch文本
const processText = (n1, n2, container) => {
  if (n1 === null) {
    // 初始化文本
    const el = (n2.el = hostCreateText(n2.children))
    hostInsert(el, container)
  } else {
    // patch文本,文本的內容變化了,我可以復用老的節點
    const el = (n2.el = n1.el)
    if (n1.children !== n2.children) {
      hostSetText(el, n2.children) // 文本的更新
    }
  }
}
  1. 處理元素類型的虛擬節點
// 處理元素,初始化元素和patch元素
const processElement = (n1, n2, container) => {
  if (n1 === null) {
    // 初始化元素
    mountElement(n2, container)
  } else {
    // patch元素
    patchElement(n1, n2)
  }
}

// 對比元素打補丁,先復用節點、再比較屬性、再比較兒子
const patchElement = (n1, n2) => {
  let el = (n2.el = n1.el)
  let oldProps = n1.props || {} // 對象
  let newProps = n2.props || {} // 對象

  patchProps(oldProps, newProps, el)
  patchChildren(n1, n2, el)
}

// 對比屬性打補丁
const patchProps = (oldProps, newProps, el) => {
  for (let key in newProps) {
    // 新的里面有,直接用新的蓋掉即可
    hostPatchProp(el, key, oldProps[key], newProps[key])
  }
  for (let key in oldProps) {
    // 如果老的里面有新的沒有,則是刪除
    if (newProps[key] == null) {
      hostPatchProp(el, key, oldProps[key], undefined)
    }
  }
}

const patchChildren = (n1, n2, el) => {
  // 核心Diff算法
}

子元素比較情況

新兒子 舊兒子 操作方式
文本 數組 (刪除老兒子,更新文本內容)
文本 文本 (更新文本內容)
文本 (更新文本內容)
數組 數組 (diff算法)
數組 文本 (清空文本,掛載元素)
數組 (掛載元素)
數組 (刪除所有子節點)
文本 (清空文本)
(不做任何處理)
// 刪除所有的子節點
const unmountChildren = children => {
  for (let i = 0; i < children.length; i++) {
    unmount(children[i])
  }
}

// 對比子節點打補丁   el: 虛擬節點對應的真實DOM元素
const patchChildren = (n1, n2, el) => {
  const c1 = n1.children
  const c2 = n2.children
  const prevShapeFlag = n1.shapeFlag // 之前的
  const shapeFlag = n2.shapeFlag // 之后的

  if (shapeFlag & ShapeFlags.TEXT_CHILDREN) {
    if (prevShapeFlag & ShapeFlags.ARRAY_CHILDREN) {
      // 文本	數組	(刪除所有子節點,更新文本內容)
      unmountChildren(c1)
    }
    if (c1 !== c2) {
      // 文本	文本	| 文本 空 (更新文本內容)
      hostSetElementText(el, c2)
    }
  } else if (shapeFlag & ShapeFlags.ARRAY_CHILDREN) {
    if (prevShapeFlag & ShapeFlags.ARRAY_CHILDREN) {
      // 數組 數組 (diff算法;全量比對)
      patchKeyedChildren(c1, c2, el)
    } else {
      if (prevShapeFlag & ShapeFlags.TEXT_CHILDREN) {
        // 數組 文本  清空文本,掛載元素)
        hostSetElementText(el, '')
      }
      // 數組 文本 | 數組 空 (清空文本,掛載元素)
      mountChildren(c2, el)
    }
  } else {
    // 空 數組
    if (prevShapeFlag & ShapeFlags.ARRAY_CHILDREN) {
      unmountChildren(c1)
    } else if (prevShapeFlag & ShapeFlags.TEXT_CHILDREN) {
      // 空 文本
      hostSetElementText(el, '')
    }
  }
}

核心Diff算法

前序比對、后序比對、同序列加掛載、同序列加卸載的目的都是:盡可能減少后面亂序比對的元素

在正式介紹diff算法之前,我們先了解幾個問題

  1. 如何判斷是否是相同的虛擬節點?

    答:虛擬節點的 type類型相同 && key相同 即可

  2. c1、c2 指的是什么?

    答:patch對比元素打補丁,先復用節點、再比較屬性、最后比較兒子節點。c1指的是舊的兒子節點;c2指的是新的兒子節點

  3. e1、e2 指的是什么?

    答:尾指針,初始值分別指向新舊孩子的最后一個節點,e1 = c1.length - 1 ;e2 = c2.length - 1

sync from start 前序對比

從頭部開始正序比對

如果是相同的虛擬節點,則調用patch對比元素打補?。ㄏ葟陀霉濣c、再比較屬性、再遞歸比較子節點),i+1

終止條件:新舊虛擬節點不一致,或, 雙方有一方 i 大于 尾指針,停止循環

 h('div',[
     h('li', { key: 'a' }, 'a'),
     h('li', { key: 'b' }, 'b'),
     h('li', { key: 'c' }, 'c')
 ]) : 
 h('div',[
     h('li', { key: 'a' }, 'a'),
     h('li', { key: 'b' }, 'b'),
     h('li', { key: 'd' }, 'd'),
     h('li', { key: 'e' }, 'e'),
     h('li', { key: 'f' }, 'f'),
 ])
// diff算法;全量比對,比較兩個兒子數組的差異
const patchKeyedChildren = (c1, c2, container) => {
  let i = 0
  // 結尾位置
  let e1 = c1.length - 1
  let e2 = c2.length - 1

  // 特殊處理,盡可能減少比對元素
  // sync from start 從頭部開始處理 O(n)
  // (a b) c
  // (a b) d e
  while (i <= e1 && i <= e2) {
    // 有任何一方停止循環則直接跳出
    const n1 = c1[i]
    const n2 = c2[i]
    // vnode type相同 && key相同
    if (isSameVnode(n1, n2)) {
      patch(n1, n2, container) // 這樣做就是比較兩個節點的屬性和子節點
    } else {
      break
    }
    i++
  }
}

sync from end 后序對比

從尾部開始倒序比對

如果是相同的虛擬節點,則調用patch對比元素打補?。ㄏ葟陀霉濣c、再比較屬性、再遞歸比較子節點),e1-1,e2-1

終止條件:新舊虛擬節點不一致,或, 雙方有一方 i 大于 尾指針,停止循環

// sync from end 從尾部開始處理 O(n)
//   a (b c)
// d e (b c)
while (i <= e1 && i <= e2) {
  const n1 = c1[e1]
  const n2 = c2[e2]
  if (isSameVnode(n1, n2)) {
    patch(n1, n2, container)
  } else {
    break
  }
  e1--
  e2--
}

common sequence+mount 同序列加掛載

分為 頭部掛載 和 尾部掛載 兩種場景

i 比 e1 大說明有要新增的,i 和 e2 之間的是新增的節點


// common sequence + mount 同序列加掛載
// i要比e1大說明有新增的;i和e2之間的是新增的部分
// (a b c)
// (a b c) d e
//     (a b c)
// e d (a b c)
if (i > e1) {
  if (i <= e2) {
    while (i <= e2) {
      const nextPos = e2 + 1
      // 根據下一個人的索引來看參照物
      const anchor = nextPos < c2.length ? c2[nextPos].el : null
      patch(null, c2[i], container, anchor) // 創建新節點 扔到容器中
      i++
    }
  }
}

common sequence+unmount 同序列加卸載

分為 頭部卸載 和 尾部卸載 兩種場景

i 比 e2 大說明有要卸載的,i 到 e1 之間的就是要卸載的節點


// common sequence + unmount 同序列加卸載
// i比e2大說明有要卸載的;i到e1之間的就是要卸載的
// (a b c) d e
// (a b c)
// e d (a b c)
//     (a b c)
else if (i > e2) {
  if (i <= e1) {
    while (i <= e1) {
      unmount(c1[i])
      i++
    }
  }
}

unknown sequence 亂序對比

keyToNewIndexMap:新節點中 key -> newIndex 的 Map 映射表,子元素中如果存在相同的 key 或者 有多個子元素沒有 key,值會被后面的索引覆蓋

newIndexToOldIndexMap:記錄新節點是否被比對過的數組映射表,初始值均為0。作用:已對比過的新節點只需移動位置;未對比過的新節點則需新創建

  1. 遍歷老的亂序節點,看一下其是否存在于新的亂序節點里面,判斷條件:keyToNewIndexMap.get(oldChild.key)是否有值,如果有則 patch 比較差異;如果沒有則要刪除老節點

  2. 在 patch 比較差異之前,我們要給 newIndexToOldIndexMap[newIndex - s2] 賦值,只是亂序比對的話,賦值為1打個標識即可。這里我們賦值為相同key的老節點索引+1,即newIndexToOldIndexMap[newIndex - s2] = i + 1(這里可能有點繞,結合代碼理解)。用于后續的最長遞增子序列

  3. 到這只是進行了新老屬性、兒子的比對 和 多余老節點的卸載操作,下面我們再執行節點位置移動和多余新節點的掛載操作

  4. 倒序遍歷新的亂序節點,看一下新節點是否被比對過,判斷條件: newIndexToOldIndexMap[i] 值是否存在非0值,如果不為0則只需移動節點位置;如果為0則要創建掛載新節點

  5. 那么移動節點的具體位置和掛載新節點的位置如何去計算?判斷條件:當前節點是否是最后一個新的子節點,let anchor = index + 1 < c2.length ? c2[index + 1].el : null,若當前節點是最后一個新的子節點,則 anchor 為 null,即插入到容器最后面;否則 anchor 為當前節點的下一個節點,即插入到 anchor 節點之前

// 優化完畢************************************
// unknown sequence 亂序比對
// (a b) 【c d e w】 (f g)
// (a b) 【e d c v】 (f g)
let s1 = i
let s2 = i
const keyToNewIndexMap = new Map() // 新節點中 key -> newIndex 的 Map 映射表,子元素中如果存在相同的key 或者 有多個子元素沒有key,值會被后面的索引覆蓋
for (let i = s2; i <= e2; i++) {
  keyToNewIndexMap.set(c2[i].key, i)
}

const toBePatched = e2 - s2 + 1 // 新的節點中亂序比對總個數
// 一個記錄是否比對過的數組映射表,作用:已對比過的節點需移動位置;未對比過的節點需新創建
const newIndexToOldIndexMap = new Array(toBePatched).fill(0)

// 遍歷老的亂序節點 看一下其是否存在于新的亂序節點里面,如果有則patch比較差異;如果沒有則要刪除老節點
for (let i = s1; i <= e1; i++) {
  const oldChild = c1[i] // 老的節點
  let newIndex = keyToNewIndexMap.get(oldChild.key) // 用老的節點去新的里面找
  if (newIndex == undefined) {
    unmount(oldChild) // 多余老節點刪掉
  } else {
    // 新的位置對應的老的位置,如果數組里放的值 >0 說明已經pactch過了。+1的目的:防止i為0
    newIndexToOldIndexMap[newIndex - s2] = i + 1 // 用來標記當前亂序新節點索引 對應的 全部老節點的加1后的索引,最長遞增子序列會用到
    patch(oldChild, c2[newIndex], container)
  }
} // 到這只是新老屬性和兒子的比對 和 多余老節點卸載操作,沒有移動位置

// 亂序節點需要移動位置,倒序遍歷亂序節點
for (let i = toBePatched - 1; i >= 0; i--) {
  let index = i + s2 // i是亂序節點中的index,需要加上s2代表全部新節點中的index
  let current = c2[index] // 找到當前節點
  let anchor = index + 1 < c2.length ? c2[index + 1].el : null
  if (newIndexToOldIndexMap[i] === 0) {
    // 創建新元素
    patch(null, current, container, anchor)
  } else {
    // 不是0,說明已經執行過patch操作了
    hostInsert(current.el, container, anchor)
  }
  // 目前無論如何都做了一遍倒敘插入,性能浪費,可以根據剛才的數組newIndexToOldIndexMap來減少插入次數
  // 用最長遞增子序列來實現,vue3新增算法,vue2在移動元素的時候則會有性能浪費
}

目前無論如何都做了一遍倒敘插入,性能浪費。

思考一下!我們是不是可以只移動一部分節點。既減少了移動次數,又能保證節點順序是正確的呢?

例如舊節點 1, 3, 4, 2,新節點 1, 2, 3, 4。那我們完全可以只將 2 移動到 3 前面,只需移動一次!就能保證順序是正確的!??!

可以用 vue3 新增算法 - 最長遞增子序列 來實現,下一篇文章吃透透?。。?/p>

總結

以上是生活随笔為你收集整理的【源码系列#06】Vue3 Diff算法的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。