成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

簡(jiǎn)化until封裝watch常用邏輯代碼

3403771864 / 405人閱讀

  有對(duì)回調(diào)進(jìn)行控制的watchWithFilter,有適用于當(dāng)watch的值為真值時(shí)觸發(fā)回調(diào)的whenever,還有只觸發(fā)一次的watchOnce和最多觸發(fā)一定次數(shù)的watchAtMost。怎么樣?是不是很多相似場(chǎng)景都有用到,主要是被觀察的變量在滿足某個(gè)具體條件時(shí)則觸發(fā)回調(diào),本篇文章until就是直到滿足某種條件時(shí)則觸發(fā)一次回調(diào)函數(shù)。我們直接看代碼。

  1.示例

  關(guān)于demo代碼:

  <script setup>
  import { until , invoke } from '@vueuse/core'
  import {ref} from 'vue'
  const source = ref(0)
  invoke(async () => {
  await until(source).toBe(4)
  console.log('滿足條件了')
  })
  const clickedFn = () => {
  source.value ++
  }
  </script>
  <template>
  <div>{{source}}</div>
  <button @click="clickedFn">
  點(diǎn)擊按鈕
  </button>
  </template>

  如上代碼所示,規(guī)定了當(dāng)source的值為4的時(shí)候觸發(fā)執(zhí)行watch回調(diào)函數(shù)。說(shuō)到invoke使用方法,就看如下代碼:

  export function invoke<T>(fn: () => T): T {
  return fn()
  }

  給定參數(shù)fn為一個(gè)函數(shù),invoke返回函數(shù)的執(zhí)行結(jié)果。代碼運(yùn)行效果如下圖所示:

1.jpg

  當(dāng)點(diǎn)擊次數(shù)達(dá)到4次時(shí),打印了相應(yīng)的信息。

  2.源碼

  until代碼較多,先看兩張預(yù)覽圖,具體代碼如下:

2.jpg

3.jpg

  通過(guò)以上兩張圖片展示出until內(nèi)部定義了很多的用于判斷條件是否滿足的方法,結(jié)果就是返回的instance也含對(duì)象。

  2.1 toMatch 

 function toMatch(
  condition: (v: any) => boolean,
  { flush = 'sync', deep = false, timeout, throwOnTimeout }: UntilToMatchOptions = {},
  ): Promise<T> {
  let stop: Function | null = null
  const watcher = new Promise<T>((resolve) => {
  stop = watch(
  r,
  (v) => {
  if (condition(v) !== isNot) {
  stop?.()
  resolve(v)
  }
  },
  {
  flush,
  deep,
  immediate: true,
  },
  )
  })
  const promises = [watcher]
  if (timeout != null) {
  promises.push(
  promiseTimeout(timeout, throwOnTimeout)
  .then(() => unref(r))
  .finally(() => stop?.()),
  )
  }
  return Promise.race(promises)
  }

  在promise構(gòu)造函數(shù)的參數(shù)函數(shù)中調(diào)用watch API來(lái)監(jiān)聽(tīng)數(shù)據(jù)源r 。當(dāng)數(shù)據(jù)源r的新值代入到條件condition中,使得condition為true時(shí)則調(diào)用stop停止監(jiān)聽(tīng)數(shù)據(jù)源,并將promise狀態(tài)變?yōu)槌晒Α?/p>

  promise放入promises數(shù)組中,如果用戶傳了timeout選項(xiàng)則promises放入調(diào)用promiseTimeout返回的promise實(shí)例。最后返回的是Promise.race的結(jié)果??匆幌聀romiseTimeout的代碼:

  export function promiseTimeout(
  ms: number,
  throwOnTimeout = false,
  reason = 'Timeout',
  ): Promise<void> {
  return new Promise((resolve, reject) => {
  if (throwOnTimeout)
  setTimeout(() => reject(reason), ms)
  else
  setTimeout(resolve, ms)
  })
  }

  promiseTimeout返回了一個(gè)promise, 如果throwOnTimeout為true則過(guò)ms毫秒之后則將promise變?yōu)槭顟B(tài),否則經(jīng)過(guò)ms毫秒后調(diào)用resolve,使promise變?yōu)槌晒顟B(tài)。

  2.2 toBe

  function toBe<P>(value: MaybeRef<P | T>, options?: UntilToMatchOptions) {
  if (!isRef(value))
  return toMatch(v => v === value, options)
  const { flush = 'sync', deep = false, timeout, throwOnTimeout } = options ?? {}
  let stop: Function | null = null
  const watcher = new Promise<T>((resolve) => {
  stop = watch(
  [r, value],
  ([v1, v2]) => {
  if (isNot !== (v1 === v2)) {
  stop?.()
  resolve(v1)
  }
  },
  {
  flush,
  deep,
  immediate: true,
  },
  )
  })
  // 和toMatch相同部分省略
  }

  toBe方法體大部分和toMatch相同,只是watch回調(diào)函數(shù)不同。這里對(duì)數(shù)據(jù)源r和toBe的參數(shù)value進(jìn)行監(jiān)聽(tīng),當(dāng)r的值和value的值相同時(shí),使promise狀態(tài)為成功。注意這里的watch使用的是偵聽(tīng)多個(gè)源的情況。

  2.3 toBeTruthy、toBeNull、toBeUndefined、toBeNaN 

 function toBeTruthy(options?: UntilToMatchOptions) {
  return toMatch(v => Boolean(v), options)
  }
  function toBeNull(options?: UntilToMatchOptions) {
  return toBe<null>(null, options)
  }
  function toBeUndefined(options?: UntilToMatchOptions) {
  return toBe<undefined>(undefined, options)
  }
  function toBeNaN(options?: UntilToMatchOptions) {
  return toMatch(Number.isNaN, options)
  }

  toBeTruthy和toBeNaN是對(duì)toMatch的封裝,toBeNull和toBeUndefined是對(duì)toBe的封裝。toBeTruthy判斷是否為真值,方法是使用Boolean構(gòu)造函數(shù)后判斷參數(shù)v是否為真值。

  toBeNaN判斷是否為NAN, 使用的是Number的isNaN作為判斷條件,注意toBeNaN的實(shí)現(xiàn)不能使用toBe, 因?yàn)閠obe在做比較的時(shí)候使用的是 ‘===’這對(duì)于NaN是不成立的:

4.jpg

  toBeNull用于判斷是否為null,toBeUndefined用于判斷是否為undefined。

  2.4 toContains

  function toContains(
  value: any,
  options?: UntilToMatchOptions,
  ) {
  return toMatch((v) => {
  const array = Array.from(v as any)
  return array.includes(value) || array.includes(unref(value))
  }, options)
  }

  判斷數(shù)據(jù)源v中是否有value,Array.from把v轉(zhuǎn)換為數(shù)組,然后使用includes方法判斷array中是否包含value。

  2.5 changed和changedTimes

 

 function changed(options?: UntilToMatchOptions) {
  return changedTimes(1, options)
  }
  function changedTimes(n = 1, options?: UntilToMatchOptions) {
  let count = -1 // skip the immediate check
  return toMatch(() => {
  count += 1
  return count >= n
  }, options)
  }

  changed用于判斷是否改變,通過(guò)調(diào)用changedTimes和固定第一參數(shù)n為1實(shí)現(xiàn)的。changedTimes的第一個(gè)參數(shù)為監(jiān)聽(tīng)的數(shù)據(jù)源改變的次數(shù),也是通過(guò)調(diào)用toMatch實(shí)現(xiàn)的,傳給toMatch的條件是一個(gè)函數(shù),此函數(shù)會(huì)在數(shù)據(jù)源改變時(shí)調(diào)用。每調(diào)用一次外層作用域定義的count就會(huì)累加一次 ,注意外層作用域count變量聲明為-1, 因?yàn)闀r(shí)立即監(jiān)聽(tīng)的。

  至此,until源碼內(nèi)定義的函數(shù)全部分析完畢,下圖總結(jié)了這些函數(shù)之前的調(diào)用關(guān)系:

5.jpg

  現(xiàn)在我們就要說(shuō)說(shuō)源碼中最終的返回值。

  2.6 until返回值——instance

  until的返回值分為兩種情況:監(jiān)聽(tīng)結(jié)果是數(shù)組時(shí)和不是數(shù)組時(shí),代碼如下圖所示: 

 if (Array.isArray(unref(r))) {
  const instance: UntilArrayInstance<T> = {
  toMatch,
  toContains,
  changed,
  changedTimes,
  get not() {
  isNot = !isNot
  return this
  },
  }
  return instance
  }
  else {
  const instance: UntilValueInstance<T, boolean> = {
  toMatch,
  toBe,
  toBeTruthy: toBeTruthy as any,
  toBeNull: toBeNull as any,
  toBeNaN,
  toBeUndefined: toBeUndefined as any,
  changed,
  changedTimes,
  get not() {
  isNot = !isNot
  return this
  },
  }
  return instance
  }

  得到的數(shù)據(jù)源時(shí)數(shù)組其中沒(méi)有toBeTruthy,toBeNull,toBeNaN,toBeUndefined這些用于判斷基本類型值的方法。另外需要注意的是返回的instance里面有一個(gè)get not(){// ...}這是使用getters, 用于獲取特定的屬性(這里是not)。在getter里面對(duì)isNot取反,isNot返回值為this也就是instance本身,所以讀取完not屬性后可以鏈?zhǔn)秸{(diào)用其他方法,如下所示:

  await until(ref).not.toBeNull()
  await until(ref).not.toBeTruthy()

  3.總結(jié)

  until方法主要是對(duì)數(shù)據(jù)監(jiān)聽(tīng),在返回時(shí),有多個(gè)具有多個(gè)條件判斷函數(shù)的對(duì)象,當(dāng)然設(shè)計(jì)者主要是可以將條件做為這些函數(shù)的參數(shù),當(dāng)監(jiān)聽(tīng)的數(shù)據(jù)滿足條件則停止監(jiān)聽(tīng),其本質(zhì)是對(duì)watch的回調(diào)進(jìn)行封裝,并結(jié)合promise.race的一個(gè)異步方法。有關(guān)demo代碼已經(jīng)上傳至github, 歡迎您親自實(shí)踐。

        本篇內(nèi)容已講述好了,大家要仔細(xì)學(xué)習(xí)。

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/128227.html

相關(guān)文章

  • 談?wù)凱ython協(xié)程技術(shù)的演進(jìn)

    摘要:事件循環(huán)是異步編程的底層基石。對(duì)事件集合進(jìn)行輪詢,調(diào)用回調(diào)函數(shù)等一輪事件循環(huán)結(jié)束,循環(huán)往復(fù)。協(xié)程直接利用代碼的執(zhí)行位置來(lái)表示狀態(tài),而回調(diào)則是維護(hù)了一堆數(shù)據(jù)結(jié)構(gòu)來(lái)處理狀態(tài)。時(shí)代的協(xié)程技術(shù)主要是,另一個(gè)比較小眾。 Coding Crush Python開(kāi)發(fā)工程師 主要負(fù)責(zé)豈安科技業(yè)務(wù)風(fēng)險(xiǎn)情報(bào)系統(tǒng)redq。 引言 1.1. 存儲(chǔ)器山 存儲(chǔ)器山是 Randal Bryant 在《深入...

    zhiwei 評(píng)論0 收藏0
  • Redis-py官方文檔翻譯

    摘要:采取兩種實(shí)現(xiàn)命令其一類盡量堅(jiān)持官方語(yǔ)法,但是以下除外沒(méi)有實(shí)現(xiàn),應(yīng)該是線程安全的原因。線程安全性是線程安全的。由于線程安全原因,不提供實(shí)現(xiàn),因?yàn)樗鼤?huì)導(dǎo)致數(shù)據(jù)庫(kù)的切換。 官網(wǎng):https://github.com/andymccurd...當(dāng)前版本:2.10.5注:這不是完整翻譯,只提取了關(guān)鍵信息。省略了部分內(nèi)容,如lua腳本支持。 pip install redis pip instal...

    Alfred 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<