溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Vue3偵聽(tīng)器的實(shí)現(xiàn)原理是什么

發(fā)布時(shí)間:2022-08-30 14:08:54 來(lái)源:億速云 閱讀:149 作者:iii 欄目:開(kāi)發(fā)技術(shù)

這篇文章主要講解了“Vue3偵聽(tīng)器的實(shí)現(xiàn)原理是什么”,文中的講解內(nèi)容簡(jiǎn)單清晰,易于學(xué)習(xí)與理解,下面請(qǐng)大家跟著小編的思路慢慢深入,一起來(lái)研究和學(xué)習(xí)“Vue3偵聽(tīng)器的實(shí)現(xiàn)原理是什么”吧!

偵聽(tīng)響應(yīng)式對(duì)象

前面我們聊到計(jì)算屬性,它可以自動(dòng)計(jì)算并緩存響應(yīng)式數(shù)據(jù)的值。而如果我們僅需要在響應(yīng)式數(shù)據(jù)變化時(shí),執(zhí)行一些預(yù)設(shè)的操作,就可以使用watch偵聽(tīng)器。我們還是先來(lái)實(shí)現(xiàn)一個(gè)最簡(jiǎn)單的例子,然后來(lái)一點(diǎn)一點(diǎn)擴(kuò)充它。

const data = {foo: 1}
const obj = reactive(data)
watch(obj, () => {
  console.log('obj已改變')
})

在這個(gè)例子中,我們使用了watch偵聽(tīng)器,當(dāng)obj的屬性被改變時(shí),控制臺(tái)應(yīng)該會(huì)打印出obj已改變?;谇懊嫖覀儗?duì)計(jì)算屬性的實(shí)現(xiàn),這里我們已經(jīng)有了一個(gè)大概的思路。把watch視為響應(yīng)式對(duì)象的副作用函數(shù),當(dāng)響應(yīng)式對(duì)象改變時(shí),觸發(fā)執(zhí)行該副作用函數(shù)。

想要觸發(fā)副作用函數(shù),必須先收集它,還記得副作用函數(shù)是如何收集的嗎?對(duì),當(dāng)響應(yīng)式數(shù)據(jù)被get時(shí),收集副作用函數(shù)。所以首先,我們需要讓watch被響應(yīng)式對(duì)象收集到。     

function watch(getter, cb) {
  effect(
    () => getter.foo
  )
}

接著,我們還需要讓我們預(yù)設(shè)的方法被執(zhí)行。當(dāng)響應(yīng)式數(shù)據(jù)被set時(shí),觸發(fā)副作用函數(shù)。這里我們想觸發(fā)的是cb這個(gè)傳入的回調(diào)函數(shù),這里我們就又能用到實(shí)現(xiàn)計(jì)算屬性時(shí)的調(diào)度器了,當(dāng)調(diào)度器存在時(shí),set觸發(fā)的trigger會(huì)先執(zhí)行調(diào)度器中的函數(shù)。

function watch(getter, cb) {
  effect(
    () => getter.foo,
    {
      scheduler() {
        cb()
      }
    }
  )
}

Vue3偵聽(tīng)器的實(shí)現(xiàn)原理是什么

一個(gè)簡(jiǎn)單的偵聽(tīng)器已經(jīng)完成了!這里我們?yōu)榱撕?jiǎn)單,把功能寫(xiě)死了,僅支持對(duì)obj.foo的偵聽(tīng)。接下來(lái),我們就要想想,如何實(shí)現(xiàn)對(duì)響應(yīng)式對(duì)象的任意屬性進(jìn)行偵聽(tīng)?

按照前面的思路,想要實(shí)現(xiàn)對(duì)響應(yīng)式對(duì)象的任意屬性的偵聽(tīng),就需要我們get到該對(duì)象的每一個(gè)屬性,這就需要我們對(duì)響應(yīng)式對(duì)象進(jìn)行一次遞歸遍歷。

function traverse(value, seen = new Set()) { // (1)
  if(typeof value !== 'object' || value === null || seen.has(value)) return
  seen.add(value)
  for(const key in value) {
    traverse(value[key], seen)
  }
  return value
}

為了避免遞歸遍歷對(duì)象時(shí),循環(huán)引用造成的死循環(huán),我們?cè)?code>(1)處創(chuàng)建了Set,當(dāng)重復(fù)出現(xiàn)相同的對(duì)象時(shí),直接返回。

偵聽(tīng)屬性值

在Vue3中,我們不能直接偵聽(tīng)響應(yīng)式對(duì)象的屬性值。如果需要偵聽(tīng)響應(yīng)式對(duì)象的屬性值,就需要一個(gè)getter函數(shù),讓偵聽(tīng)器能被響應(yīng)式對(duì)象收集到。

const data = {
  foo: 1
}
const obj = reactive(data)
watch(
  () => obj.foo, 
  () => {
  console.log('obj.foo已改變')
})

指定了屬性就意味著,當(dāng)前的偵聽(tīng)器僅會(huì)被指定的屬性觸發(fā),就無(wú)需遞歸遍歷整個(gè)響應(yīng)式對(duì)象了。

function watch(getter, cb) {
  if(typeof getter !== 'function') getter = traverse(getter) // (2)
  effect(
    () => getter(),
    {
      scheduler() {
        cb()
      }
    }
  )
}

Vue3偵聽(tīng)器的實(shí)現(xiàn)原理是什么

在(2)處,我們?cè)黾恿艘粋€(gè)判斷,如果傳入的已經(jīng)是getter函數(shù),我們直接使用,如果不是getter函數(shù),則認(rèn)為是一個(gè)響應(yīng)式對(duì)象,就需要進(jìn)行遞歸遍歷。

偵聽(tīng)獲取新值和舊值

在Vue中我們還需要能夠在回調(diào)函數(shù)cb()中拿到響應(yīng)式數(shù)據(jù)更新前后的新值與舊值。

const data = {
  foo: 1
}
const obj = reactive(data)
watch(
  () => obj.foo, 
  (newValue, oldValue) => {
  console.log(newValue, oldValue)
})

接下來(lái)的問(wèn)題是,如何獲取newValueoldValue。newValue好解決,執(zhí)行完回調(diào)函數(shù)cb()得到的就是newValue,但這里如何獲取oldValue的值呢?要從watch中拿到舊值,那就不能讓副作用函數(shù)被立即執(zhí)行。這里想到了什么?對(duì),在實(shí)現(xiàn)計(jì)算屬性的時(shí)候,我們用到過(guò)的lazy,它可以禁止副作用函數(shù)自動(dòng)執(zhí)行。

function watch(getter, cb) {
  if(typeof getter !== 'function') getter = traverse(getter)
  let oldValue
  const effectFn = effect(
    () => getter(),
    {
      lazy: true, // (3)
      scheduler() {
          cb(oldValue)
      }
    }
  )
  oldValue = effectFn() // (4)
}

在(3)處我們?cè)O(shè)置了lazy開(kāi)關(guān),設(shè)置了lazy后,副作用函數(shù)的執(zhí)行權(quán)就交到了我們自己手上。在(4)處,我們手動(dòng)執(zhí)行了副作用函數(shù)。這里可以需要我們向前回顧一下,前面我們傳入的getter是一個(gè)函數(shù)() => obj.foo,而effect函數(shù)的第一個(gè)參數(shù)就是真正被執(zhí)行的副作用函數(shù),所以我們手動(dòng)執(zhí)行的,其實(shí)就是函數(shù)() => obj.foo,這樣我們就拿到了舊值。

如何獲取新值呢?在響應(yīng)式數(shù)據(jù)的值更新后,副作用函數(shù)effect會(huì)被觸發(fā)執(zhí)行,當(dāng)調(diào)度器屬性存在時(shí),執(zhí)行調(diào)度器。在調(diào)度器中,我們可以再次執(zhí)行副作用函數(shù),通過(guò)() => obj.foo拿到改變后的新值。

function watch(getter, cb) {
  if(typeof getter !== 'function') getter = traverse(getter)
  let oldValue, newValue
  const effectFn = effect(
    () => getter(),
    {
      lazy: true,
      scheduler() {
        newValue = effectFn()
        cb(newValue, oldValue)
        oldValue = newValue // (5)
      }
    }
  )
  oldValue = effectFn()
}

在(5)處,執(zhí)行完回調(diào)函數(shù)cb(),我們進(jìn)行了一下善后工作,更新了oldValue的值,為下一次回調(diào)做準(zhǔn)備。

Vue3偵聽(tīng)器的實(shí)現(xiàn)原理是什么

有時(shí),我們還希望偵聽(tīng)器可以在創(chuàng)建時(shí)就立即執(zhí)行回調(diào)函數(shù)。

const data = {
  foo: 1
}
const obj = reactive(data)
watch(
  () => obj.foo, 
  (newValue, oldValue) => {
      console.log('newValue:', newValue,', oldValue:', oldValue)
  },
  { immediate: true }
)

當(dāng)immediate的值為true時(shí),需要立即執(zhí)行。明確了需求,我們來(lái)完善watch偵聽(tīng)器。

function watch(getter, cb, options = {}) {
  if(typeof getter !== 'function') getter = traverse(getter)
  let oldValue, newValue
  function job() { // (6)
    newValue = effectFn()
    cb(newValue, oldValue)
    oldValue = newValue
  }

  const effectFn = effect(
    () => getter(),
    {
      lazy: true,
      scheduler: job,
    }
  )

  if(options.immediate) {  // (7)
    job()
  } else {
    oldValue = effectFn()
  } 
}

在(6)處,我們抽離了回調(diào)函數(shù)的執(zhí)行邏輯,當(dāng)options.immediate存在時(shí),直接觸發(fā)執(zhí)行。

Vue3偵聽(tīng)器的實(shí)現(xiàn)原理是什么

實(shí)現(xiàn)效果

Vue3偵聽(tīng)器的實(shí)現(xiàn)原理是什么

感謝各位的閱讀,以上就是“Vue3偵聽(tīng)器的實(shí)現(xiàn)原理是什么”的內(nèi)容了,經(jīng)過(guò)本文的學(xué)習(xí)后,相信大家對(duì)Vue3偵聽(tīng)器的實(shí)現(xiàn)原理是什么這一問(wèn)題有了更深刻的體會(huì),具體使用情況還需要大家實(shí)踐驗(yàn)證。這里是億速云,小編將為大家推送更多相關(guān)知識(shí)點(diǎn)的文章,歡迎關(guān)注!

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI