欧美一级特黄大片做受成人-亚洲成人一区二区电影-激情熟女一区二区三区-日韩专区欧美专区国产专区

Vue3偵聽器怎么應(yīng)用

本篇內(nèi)容介紹了“Vue3偵聽器怎么應(yīng)用”的有關(guān)知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領(lǐng)大家學(xué)習(xí)一下如何處理這些情況吧!希望大家仔細閱讀,能夠?qū)W有所成!

創(chuàng)新互聯(lián)建站堅持“要么做到,要么別承諾”的工作理念,服務(wù)領(lǐng)域包括:成都網(wǎng)站建設(shè)、網(wǎng)站設(shè)計、企業(yè)官網(wǎng)、英文網(wǎng)站、手機端網(wǎng)站、網(wǎng)站推廣等服務(wù),滿足客戶于互聯(lián)網(wǎng)時代的宕昌網(wǎng)站設(shè)計、移動媒體設(shè)計的需求,幫助企業(yè)找到有效的互聯(lián)網(wǎng)解決方案。努力成為您成熟可靠的網(wǎng)絡(luò)建設(shè)合作伙伴!

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

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

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

在這個例子中,我們使用了watch偵聽器,當(dāng)obj的屬性被改變時,控制臺應(yīng)該會打印出obj已改變。基于前面我們對計算屬性的實現(xiàn),這里我們已經(jīng)有了一個大概的思路。把watch視為響應(yīng)式對象的副作用函數(shù),當(dāng)響應(yīng)式對象改變時,觸發(fā)執(zhí)行該副作用函數(shù)。

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

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

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

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

Vue3偵聽器怎么應(yīng)用

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

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

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,當(dāng)重復(fù)出現(xiàn)相同的對象時,直接返回。

偵聽屬性值

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

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

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

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

Vue3偵聽器怎么應(yīng)用

在(2)處,我們增加了一個判斷,如果傳入的已經(jīng)是getter函數(shù),我們直接使用,如果不是getter函數(shù),則認為是一個響應(yī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)
})

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

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

Vue3偵聽器怎么應(yīng)用

有時,我們還希望偵聽器可以在創(chuàng)建時就立即執(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時,需要立即執(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)處,我們抽離了回調(diào)函數(shù)的執(zhí)行邏輯,當(dāng)options.immediate存在時,直接觸發(fā)執(zhí)行。

Vue3偵聽器怎么應(yīng)用

實現(xiàn)效果

Vue3偵聽器怎么應(yīng)用

“Vue3偵聽器怎么應(yīng)用”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識可以關(guān)注創(chuàng)新互聯(lián)網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實用文章!

文章題目:Vue3偵聽器怎么應(yīng)用
分享路徑:http://aaarwkj.com/article22/iihgcc.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站維護、標(biāo)簽優(yōu)化、靜態(tài)網(wǎng)站、網(wǎng)站策劃、移動網(wǎng)站建設(shè)、響應(yīng)式網(wǎng)站

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

成都做網(wǎng)站
91精品人妻一区二区三区| av成熟一区二区三区| 亚洲av产在线精品亚洲第| 女性裸体无遮挡啪啪网站| 成人精品播放视频在线观看| av在线免费观看青青草原| 91日本精品一区二区| 国产真人免费作爱视频网站 | 免费观看国产裸体视频| 免费的黄色片带中文字幕| 日韩欧美国产精品自拍| 国产精品粉嫩在线播放| 高清日本一区二区三区不卡片| 青青草免费视频观看在线| 国产日韩在线不卡网站| 丰满人妻大屁一区二区| 男女午夜激情啪啪视频| 日韩欧美的一区二区三区| 欧美高清视频免费播放| 国产av一区二区三区日韩接吻| 中文字幕乱码人妻一区| 日韩精品人妻一区二区网站| 亚洲日本成人一区二区| 日韩成人在线高清视频| 美味人妻手机在线观看| 2022国产精品一区| 亚洲乱色熟女一区二区三区麻豆| 青青草免费在线视频视频| 有码国内精品人妻少妇| 国产亚洲一线二线三线| 精品人妻av区天天看片| 伊人激情久久综合中文字幕| 免费国产污在线观看网站| 欧美精品在,欧美一区二区| 亚洲第一女人天堂av| 国产精品亚洲欧美中字| 亚洲精品日本一区二区| 日日添夜夜添天天操| 97门久欧美日韩久久| 国产精品亚洲二区三区| 小草少妇视频免费看视频|