资讯专栏INFORMATION COLUMN

防抖 - 理解,实践与实现

yangrd / 2025人阅读

摘要:本文主要讨论防抖,镜像文章节流理解,实践与实现。分开讨论防抖和节流,主要是为了让一些还不太了解防抖节流的读者能够有针对性地,逐一掌握它们。防抖是什么结合上方案例,防抖可以理解为多次触发事件后,事件处理函数只执行一次。

为了完整阅读体验,欢迎移步到我的博客原文。

防抖(去抖),以及节流(分流)在日常开发中可能用的不多,但在特定场景,却十分有用。本文主要讨论防抖,镜像文章:节流 - 理解,实践与实现。分开讨论防抖和节流,主要是为了让一些还不太了解防抖节流的读者能够有针对性地,逐一掌握它们。
防抖有两种模式(容易让人迷惑):延时执行和直接执行。后续详细讨论。
防抖还有一个关键点是如果用代码实现。本文以循序渐进地方式,先以实现一个简单案例为例,绘制流程图,再根据流程图逻辑编写防抖功能代码。

典型案例

以日常开发中常用的搜索按钮为例,若用户点击一次搜索按钮后,不小心“手抖”很快又点了一次按钮,防抖可以规避第二次甚至更多次搜索。


点击查看案例

第一个搜索按钮未做任何防抖处理。
搜索按钮A为第一种防抖模式:延时执行。若用户连续快速点击多次,只有最后一次点击结束,延时一段时间后才执行搜索。
搜索按钮B为第二种防抖模式:直接执行。若用户连续快速点击多次,只有第一次点击会执行搜索。

防抖是什么

结合上方案例,防抖可以理解为:多次触发事件后,事件处理函数只执行一次。
而防抖的两种模式可以根据实际使用场景分别应用。

应用场景

在搜索框中实时键入文本搜索

点击查看案例

防止频繁点击搜索按钮触发搜索请求

点击查看案例

一步步手写防抖

接下来我们通过一个案例梳理实现防抖的思路。
假设我们要实现本文第一个案例中搜索按钮A的功能。首先整理需求:

点击搜索按钮后,函数并不马上执行,而是等一段时间再执行。

若在这段时间内,按钮再次被点击,则重新开始计时,等待同样一段时间后再执行。

实现的方法有两种,推荐第一种,用计时器(setTimeout)简化代码,将重心放在实现防抖的逻辑上。

方法一核心参数:

等待时长

计时器

绘制方法一的流程图:

根据流程图思路实现方法一的防抖代码:

function debounce( func, wait = 0 ) {
  let timer
  
  function debounced( ...args ) {
    const self = this
    if ( timer == null ) {
      addTimer()
      return
    }

    if ( timer != null ) {
      clearTimer()
      addTimer()
      return
    }

    function addTimer() {
      timer = setTimeout( () => {
        invokeFunc()
        clearTimer()
      }, wait )
    }

    function invokeFunc() {
      func.apply( self, args )
    }
  }

  return debounced

  function clearTimer() {
    clearTimeout( timer )
    timer = null
  }
}

方法二核心参数:

等待时长

最早可执行时间

绘制方法二的流程图:

根据流程图实现方法二的防抖代码:

function debounce( func, wait = 0 ) {
  // Earliest time when func can be invoked
  let earliest
  
  function debounced( ...args ) {
    const self = this
    
    if ( typeof earliest === "undefined" ) {
      setEarliset()
    }

    if ( typeof earliest !== "undefined" ) {
      if ( now() >= earliest ) {
        invokeFun()
      } else {
        setEarliset()
      }
    }

    function setEarliset() {
      earliest = now() + wait
    }

    function invokeFun() {
      func.apply( self, args )
    }
  }

  return debounced

  function now() {
    return +new Date()
  }
}

同样,我们可以使用类似方法实现搜索按钮B的功能。
需求描述:

点击搜索按钮后,函数马上执行。只有等待一段时间后被点击才能执行函数。

若在这段时间内按钮被点击,则重新计时。

核心参数:

等待时长

计时器

function debounce( func, wait = 0 ) {
  let timer
  
  function debounced( ...args ) {
    const self = this

    timer == null && invokeFunc()

    timer != null && clearTimer()
    
    timer = setTimeout( clearTimer, wait )

    function invokeFunc() {
      func.apply( self, args )
    }
  }

  return debounced

  function clearTimer() {
    clearTimeout( timer )
    timer = null
  }
}

接下来我们使用刚才编写的debounce函数来测试第一个案例

点击查看案例

总结

防抖是一个高阶函数,能够将多个事件函数合并为一个,在“调整window尺寸”,“在搜索框中实时搜索键入文本”, “滚动滚动条”和“防止搜索按钮频繁点击触发多余请求”等案例中,十分有用。

链接

Lodash推荐:https://css-tricks.com/deboun...

简单理解防抖定义:https://www.cnblogs.com/woody...

感谢你花时间阅读这篇文章。如果你喜欢这篇文章,欢迎点赞、收藏和分享,让更多的人看到这篇文章,这也是对我最大的鼓励和支持!
同时欢迎阅读我的更多原创前端技术博客: 苏溪云的博客。

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/108521.html

相关文章

  • 节流 - 理解实践实现

    摘要:节流分流,与防抖去抖实现原理相似。本文主要讨论节流,镜像文章防抖理解,实践与实现。分开讨论防抖和节流,主要是为了让一些还不太了解节流防抖的读者能够有针对性地,逐一掌握它们。上方为未节流模式,每一次触发都会绘制一个圆点。 showImg(https://segmentfault.com/img/bVbqMwN?w=1280&h=720); 节流(分流),与防抖(去抖)实现原理相似。本文主...

    layman 评论0 收藏0
  • 移动端手势库设计实践

    摘要:前言本次给大家分享的是常见的移动端单点触摸事件的设计思路及实践。实现即手指滑动事件,应用场景如轮播图左右滑动切换,整屏页面滑动翻页等,算是移动端最常见的手势之一了。 前言 本次给大家分享的是常见的移动端单点触摸事件的设计思路及实践。 核心技术 主要就是利用移动端的以下3个触摸事件,来模拟和实现自定义的手势操作 touchstart:手指触摸到屏幕的一瞬间触发 touchmove:手指...

    wudengzan 评论0 收藏0
  • 初探函数节流和函数防抖—以项目为例(更新es6语法)

    摘要:而上述两种方法,就叫做函数的节流和防抖。二函数节流和函数防抖函数节流函数节流函数节流是让这个函数在间隔某一段时间执行一次。在这个项目中,我认为函数节流和函数防抖都能很好的解决问题。 一 项目需求 最近在做一些小的练手代码的时候,碰到了一个很常见的问题,当在搜索框中进行搜索的时候,如果快速输入很多字符的话,搜索框的监听回调函数会执行很多次,如果回调业务较复杂的话,可能会导致页面运行缓慢甚...

    赵连江 评论0 收藏0
  • 「中高级前端面试」JavaScript手写代码无敌秘籍

    摘要:第一种直接调用避免在不必要的情况下使用,是一个危险的函数,他执行的代码拥有着执行者的权利。来自于此外,实现需要考虑实例化后对原型链的影响。函数柯里化的主要作用和特点就是参数复用提前返回和延迟执行。手写路径导航 实现一个new操作符 实现一个JSON.stringify 实现一个JSON.parse 实现一个call或 apply 实现一个Function.bind 实现一个继承 实现一个J...

    Zhuxy 评论0 收藏0
  • 前端进击的巨人(八):浅谈函数防抖节流

    摘要:隆重请出主角防抖与节流。防抖与节流的异同相同都是防止某一时间段内,函数被频繁调用执行,通过时间频率控制,减少回调函数执行次数,来实现相关性能优化。参考文章分钟理解的节流防抖及使用场景函数防抖和节流 showImg(https://segmentfault.com/img/bVburM8?w=800&h=600); 本篇课题,或许早已是烂大街的解读文章。不过春招系列面试下来,不少伙伴们还...

    _Zhao 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<