亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Vue3偵聽器的實現原理詳情

 更新時間:2022年08月30日 11:54:05   作者:???????咕咕雞_  
這篇文章主要介紹了Vue3偵聽器的實現原理詳情,文章圍繞主題展開詳細的內容介紹,具有一定的參考價值,需要的小伙伴可以參考一下

前言:

本篇內容基于Vue3計算屬性是如何實現的實現。

偵聽響應式對象

前面我們聊到計算屬性,它可以自動計算并緩存響應式數據的值。而如果我們僅需要在響應式數據變化時,執(zhí)行一些預設的操作,就可以使用watch偵聽器。我們還是先來實現一個最簡單的例子,然后來一點一點擴充它。

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

在這個例子中,我們使用了watch偵聽器,當obj的屬性被改變時,控制臺應該會打印出obj已改變?;谇懊嫖覀儗τ嬎銓傩缘膶崿F,這里我們已經有了一個大概的思路。把watch視為響應式對象的副作用函數,當響應式對象改變時,觸發(fā)執(zhí)行該副作用函數。

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

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

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

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

一個簡單的偵聽器已經完成了!這里我們?yōu)榱撕唵危压δ軐懰懒?,僅支持對obj.foo的偵聽。接下來,我們就要想想,如何實現對響應式對象的任意屬性進行偵聽?

按照前面的思路,想要實現對響應式對象的任意屬性的偵聽,就需要我們get到該對象的每一個屬性,這就需要我們對響應式對象進行一次遞歸遍歷。

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
}

為了避免遞歸遍歷對象時,循環(huán)引用造成的死循環(huán),我們在(1)處創(chuàng)建了Set,當重復出現相同的對象時,直接返回。

偵聽屬性值

在Vue3中,我們不能直接偵聽響應式對象的屬性值。如果需要偵聽響應式對象的屬性值,就需要一個getter函數,讓偵聽器能被響應式對象收集到。

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

指定了屬性就意味著,當前的偵聽器僅會被指定的屬性觸發(fā),就無需遞歸遍歷整個響應式對象了。

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

在(2)處,我們增加了一個判斷,如果傳入的已經是getter函數,我們直接使用,如果不是getter函數,則認為是一個響應式對象,就需要進行遞歸遍歷。

偵聽獲取新值和舊值

在Vue中我們還需要能夠在回調函數cb()中拿到響應式數據更新前后的新值與舊值。

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

接下來的問題是,如何獲取newValueoldValue。newValue好解決,執(zhí)行完回調函數cb()得到的就是newValue,但這里如何獲取oldValue的值呢?要從watch中拿到舊值,那就不能讓副作用函數被立即執(zhí)行。這里想到了什么?對,在實現計算屬性的時候,我們用到過的lazy,它可以禁止副作用函數自動執(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)處我們設置了lazy開關,設置了lazy后,副作用函數的執(zhí)行權就交到了我們自己手上。在(4)處,我們手動執(zhí)行了副作用函數。這里可以需要我們向前回顧一下,前面我們傳入的getter是一個函數() => obj.foo,而effect函數的第一個參數就是真正被執(zhí)行的副作用函數,所以我們手動執(zhí)行的,其實就是函數() => obj.foo,這樣我們就拿到了舊值。

如何獲取新值呢?在響應式數據的值更新后,副作用函數effect會被觸發(fā)執(zhí)行,當調度器屬性存在時,執(zhí)行調度器。在調度器中,我們可以再次執(zhí)行副作用函數,通過() => 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í)行完回調函數cb(),我們進行了一下善后工作,更新了oldValue的值,為下一次回調做準備。

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

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

immediate的值為true時,需要立即執(zhí)行。明確了需求,我們來完善watch偵聽器。

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)處,我們抽離了回調函數的執(zhí)行邏輯,當options.immediate存在時,直接觸發(fā)執(zhí)行。

實現效果

到此這篇關于Vue3偵聽器是如何實現的的文章就介紹到這了,更多相關Vue3偵聽器內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • vue實現目錄樹結構

    vue實現目錄樹結構

    這篇文章主要為大家詳細介紹了vue實現目錄樹結構,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-03-03
  • vue通信方式EventBus的實現代碼詳解

    vue通信方式EventBus的實現代碼詳解

    這篇文章主要介紹了vue通信方法EventBus的實現代碼,本文給大家介紹的非常詳細,具有一定的參考借鑒價值,需要的朋友可以參考下
    2019-06-06
  • vue 解決mintui彈窗彈起來,底部頁面滾動bug問題

    vue 解決mintui彈窗彈起來,底部頁面滾動bug問題

    這篇文章主要介紹了vue 解決mintui彈窗彈起來,底部頁面滾動bug問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-11-11
  • 淺談vuex中store的命名空間

    淺談vuex中store的命名空間

    今天小編就為大家分享一篇淺談vuex中store的命名空間,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2019-11-11
  • Vue如何解決子組件data從props中無法動態(tài)更新數據問題

    Vue如何解決子組件data從props中無法動態(tài)更新數據問題

    這篇文章主要介紹了Vue如何解決子組件data從props中無法動態(tài)更新數據問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-10-10
  • vue實現商城上貨組件簡易版

    vue實現商城上貨組件簡易版

    這篇文章主要為大家詳細介紹了vue實現商城上貨組件簡易版,50行js代碼實現效果,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-11-11
  • vue 在服務器端直接修改請求的接口地址

    vue 在服務器端直接修改請求的接口地址

    這篇文章主要介紹了vue 在服務器端直接修改請求的接口地址的方法,幫助大家更好的理解和使用vue,感興趣的朋友可以了解下
    2020-12-12
  • Vue3如何獲取proxy對象的值而不是引用的方式

    Vue3如何獲取proxy對象的值而不是引用的方式

    這篇文章主要介紹了Vue3如何獲取proxy對象的值而不是引用的方式,本文通過示例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2023-10-10
  • 淺析webpack-bundle-analyzer在vue-cli3中的使用

    淺析webpack-bundle-analyzer在vue-cli3中的使用

    這篇文章主要介紹了webpack-bundle-analyzer在vue-cli3中的使用,非常不錯,具有一定的參考借鑒價值,需要的朋友可以參考下
    2019-10-10
  • vue2響應式原理之Object.defineProperty()方法的使用

    vue2響應式原理之Object.defineProperty()方法的使用

    這篇文章主要介紹了vue2響應式原理之Object.defineProperty()方法的使用,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2023-10-10

最新評論