防抖而钞、節(jié)流實(shí)現(xiàn)方式

防抖、節(jié)流

開發(fā)中經(jīng)常會有一些持續(xù)觸發(fā)的事件拘荡,比如scroll臼节、resize、mousemove、input等网缝。頻繁的執(zhí)行回調(diào)巨税,不僅對性能有很大影響,甚至?xí)邢鄳?yīng)跟不上粉臊,造成頁面卡死等現(xiàn)象草添。

針對這種問題有兩種解決方案,防抖和節(jié)流扼仲。

防抖

事件觸發(fā)后的time時(shí)間內(nèi)只執(zhí)行一次果元。原理是維護(hù)一個(gè)延時(shí)器,規(guī)定在time時(shí)間后執(zhí)行函數(shù)犀盟,如果在time時(shí)間內(nèi)再次觸發(fā),則取消之前的延時(shí)器重新設(shè)置蝇狼。所以回調(diào)只在最后執(zhí)行一次阅畴。

  1. 方式一:
function debounce(func, time = 0) {
  if (typeof func !== 'function') {
    throw new TypeError('Expect a function')
  }

  let timer

  return function (e) {
    if (timer) {
      clearTimeout(timer)
    }

    timer = setTimeout(() => {
      func.call(this, e)
      clearTimeout(timer)
    }, time)
  }
}
  1. 方式二:
const debounce = (func, wait = 0) => {
  let timeout = null
  let args
  function debounced(...arg) {
    args = arg
    if(timeout) {
      clearTimeout(timeout)
      timeout = null
    }
    // 以Promise的形式返回函數(shù)執(zhí)行結(jié)果
    return new Promise((res, rej) => {
      timeout = setTimeout(async () => {
        try {
          const result = await func.apply(this, args)
          res(result)
        } catch(e) {
          rej(e)
        }
      }, wait)
    })
  }
  // 允許取消
  function cancel() {
    clearTimeout(timeout)
    timeout = null
  }
  // 允許立即執(zhí)行
  function flush() {
    cancel()
    return func.apply(this, args)
  }
  debounced.cancel = cancel
  debounced.flush = flush
  return debounced
}

節(jié)流

在事件觸發(fā)過程中,每隔wait執(zhí)行一次回調(diào)迅耘。

可選參數(shù)三 trailing贱枣,事件第一次觸發(fā)是否執(zhí)行一次回調(diào)。默認(rèn)true颤专,即第一次觸發(fā)先執(zhí)行一次纽哥。

  1. 方式一:
function throttle(func, wait = 0, trailing = true) {
  if (typeof func !== 'function') {
    throw new TypeError('Expected a function')
  }

  let timer
  let start = +new Date

  return function (e) {
    const now = +new Date
    const distance = start + wait - now

    if (timer) {
      clearTimeout(timer)
    }
    if (now - start >= wait || trailing) { // 每個(gè)wait時(shí)間執(zhí)行一次或第一次觸發(fā)就執(zhí)行一次
      func.apply(this, rest)
      start = now
      trailing = false
    } else {
      // 事件結(jié)束wait時(shí)間后再執(zhí)行一次
      timer = setTimeout(() => {
        func.call(this, e)
      }, distance )
    }
  }
}
  1. 方式二:
function throttlew(fn, wait) {
    let start = 0

    return function(e) {
        const now = Date.now()
        if (now - start > wait) {
            fn.call(this, e)
            start = now
        }
 }    

  1. 方式三:
const throttle = (func, wait = 0, execFirstCall) => {
    let timeout = null
    let args
    let firstCallTimestamp

    function throttled(...arg) {
      if (!firstCallTimestamp) firstCallTimestamp = new Date().getTime()
      if (!execFirstCall || !args) {
        // console.log('set args:', arg)
        args = arg
      }
      if (timeout) {
        clearTimeout(timeout)
        timeout = null
      }
      // 以Promise的形式返回函數(shù)執(zhí)行結(jié)果
      return new Promise(async (res, rej) => {
        if (new Date().getTime() - firstCallTimestamp >= wait) {
          try {
            const result = await func.apply(this, args)
            res(result)
          } catch (e) {
            rej(e)
          } finally {
            cancel()
          }
        } else {
          timeout = setTimeout(async () => {
            try {
              const result = await func.apply(this, args)
              res(result)
            } catch (e) {
              rej(e)
            } finally {
              cancel()
            }
          }, firstCallTimestamp + wait - new Date().getTime())
        }
      })
    }
    // 允許取消
    function cancel() {
      clearTimeout(timeout)
      args = null
      timeout = null
      firstCallTimestamp = null  // 這里很關(guān)鍵。每次執(zhí)行完成后都要初始化
    }
    // 允許立即執(zhí)行
    function flush() {
      cancel()
      return func.apply(this, args)
    }
    throttled.cancel = cancel
    throttled.flush = flush
    return throttled
  }

區(qū)別: 不管事件觸發(fā)有多頻繁栖秕,節(jié)流都會保證在規(guī)定時(shí)間內(nèi)一定會執(zhí)行一次真正的事件處理函數(shù)春塌,而防抖只是在最后一次事件觸發(fā)后才執(zhí)行一次函數(shù)。

場景:

防抖:比如監(jiān)聽頁面滾動簇捍,滾動結(jié)束并且到達(dá)一定距離時(shí)顯示返回頂部按鈕只壳,適合使用防抖。

節(jié)流:比如在頁面的無限加載場景下暑塑,需要用戶在滾動頁面時(shí)過程中吼句,每隔一段時(shí)間發(fā)一次 Ajax 請求,而不是在用戶停下滾動頁面操作時(shí)才去請求數(shù)據(jù)事格。此場景適合用節(jié)流來實(shí)現(xiàn)惕艳。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市驹愚,隨后出現(xiàn)的幾起案子远搪,更是在濱河造成了極大的恐慌,老刑警劉巖么鹤,帶你破解...
    沈念sama閱讀 218,122評論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件终娃,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)棠耕,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,070評論 3 395
  • 文/潘曉璐 我一進(jìn)店門余佛,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人窍荧,你說我怎么就攤上這事辉巡。” “怎么了蕊退?”我有些...
    開封第一講書人閱讀 164,491評論 0 354
  • 文/不壞的土叔 我叫張陵郊楣,是天一觀的道長。 經(jīng)常有香客問我瓤荔,道長净蚤,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,636評論 1 293
  • 正文 為了忘掉前任输硝,我火速辦了婚禮今瀑,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘点把。我一直安慰自己橘荠,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,676評論 6 392
  • 文/花漫 我一把揭開白布郎逃。 她就那樣靜靜地躺著哥童,像睡著了一般。 火紅的嫁衣襯著肌膚如雪褒翰。 梳的紋絲不亂的頭發(fā)上贮懈,一...
    開封第一講書人閱讀 51,541評論 1 305
  • 那天,我揣著相機(jī)與錄音优训,去河邊找鬼错邦。 笑死,一個(gè)胖子當(dāng)著我的面吹牛型宙,可吹牛的內(nèi)容都是我干的撬呢。 我是一名探鬼主播,決...
    沈念sama閱讀 40,292評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼妆兑,長吁一口氣:“原來是場噩夢啊……” “哼魂拦!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起搁嗓,我...
    開封第一講書人閱讀 39,211評論 0 276
  • 序言:老撾萬榮一對情侶失蹤芯勘,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后腺逛,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體荷愕,經(jīng)...
    沈念sama閱讀 45,655評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,846評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了安疗。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片抛杨。...
    茶點(diǎn)故事閱讀 39,965評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖荐类,靈堂內(nèi)的尸體忽然破棺而出怖现,到底是詐尸還是另有隱情,我是刑警寧澤玉罐,帶...
    沈念sama閱讀 35,684評論 5 347
  • 正文 年R本政府宣布屈嗤,位于F島的核電站,受9級特大地震影響吊输,放射性物質(zhì)發(fā)生泄漏饶号。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,295評論 3 329
  • 文/蒙蒙 一季蚂、第九天 我趴在偏房一處隱蔽的房頂上張望讨韭。 院中可真熱鬧,春花似錦癣蟋、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,894評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至埋泵,卻和暖如春幔欧,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背丽声。 一陣腳步聲響...
    開封第一講書人閱讀 33,012評論 1 269
  • 我被黑心中介騙來泰國打工礁蔗, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人雁社。 一個(gè)月前我還...
    沈念sama閱讀 48,126評論 3 370
  • 正文 我出身青樓浴井,卻偏偏與公主長得像,于是被迫代替她去往敵國和親霉撵。 傳聞我的和親對象是個(gè)殘疾皇子磺浙,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,914評論 2 355

推薦閱讀更多精彩內(nèi)容