博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
理解函数防抖Debounce
阅读量:6272 次
发布时间:2019-06-22

本文共 7809 字,大约阅读时间需要 26 分钟。

一、函数为什么要防抖

有如下代码

window.onresize = () => {  console.log('触发窗口监听回调函数')}复制代码

当我们在PC上缩放浏览器窗口时,一秒可以轻松触发30次事件。手机端触发其他Dom时间监听回调时同理。

这里的回调函数只是打印字符串,如果回调函数更加复杂,可想而知浏览器的压力会非常大,用户体验会很糟糕。

resizescroll等Dom事件的监听回调会被频繁触发,因此我们要对其进行限制。

二、实现思路

函数去抖简单来说就是对于一定时间段的连续的函数调用,只让其执行一次,初步的实现思路如下:

第一次调用函数,创建一个定时器,在指定的时间间隔之后运行代码。当第二次调用该函数时,它会清除前一次的定时器并设置另一个。如果前一个定时器已经执行过了,这个操作就没有任何意义。然而,如果前一个定时器尚未执行,其实就是将其替换为一个新的定时器。目的是只有在执行函数的请求停止了一段时间之后才执行。

三、Debounce 应用场景

  • 每次 resize/scroll 触发统计事件
  • 文本输入的验证(连续输入文字后发送 AJAX 请求进行验证,验证一次就好)

四、函数防抖最终版

代码说话,有错恳请指出

function debounce(method, wait, immediate) {  let timeout  // debounced函数为返回值  // 使用Async/Await处理异步,如果函数异步执行,等待setTimeout执行完,拿到原函数返回值后将其返回  // args为返回函数调用时传入的参数,传给method  let debounced = function(...args) {    return new Promise (resolve => {      // 用于记录原函数执行结果      let result      // 将method执行时this的指向设为debounce返回的函数被调用时的this指向      let context = this      // 如果存在定时器则将其清除      if (timeout) {        clearTimeout(timeout)      }      // 立即执行需要两个条件,一是immediate为true,二是timeout未被赋值或被置为null      if (immediate) {        // 如果定时器不存在,则立即执行,并设置一个定时器,wait毫秒后将定时器置为null        // 这样确保立即执行后wait毫秒内不会被再次触发        let callNow = !timeout        timeout = setTimeout(() => {          timeout = null        }, wait)        // 如果满足上述两个条件,则立即执行并记录其执行结果        if (callNow) {          result = method.apply(context, args)          resolve(result)        }      } else {        // 如果immediate为false,则等待函数执行并记录其执行结果        // 并将Promise状态置为fullfilled,以使函数继续执行        timeout = setTimeout(() => {          // args是一个数组,所以使用fn.apply          // 也可写作method.call(context, ...args)          result = method.apply(context, args)          resolve(result)        }, wait)      }    })  }  // 在返回的debounced函数上添加取消方法  debounced.cancel = function() {    clearTimeout(timeout)    timeout = null  }  return debounced}复制代码

需要注意的是,如果需要原函数返回值,调用防抖后的函数的外层函数需要使用Async/Await语法等待执行结果返回

使用方法见代码:

function square(num) {  return Math.pow(num, 2)}let debouncedFn = debounce(square, 1000, false)window.addEventListener('resize', async () => {  let val  try {    val = await debouncedFn(4)  } catch (err) {    console.error(err)  }  // 停止缩放1S后输出:  // 原函数的返回值为:16  console.log(`原函数返回值为${val}`)}, false)复制代码

具体的实现步骤请往下看

五、Debounce 的实现

1. 《JavaScript高级程序设计》(第三版)中的实现

function debounce(method, context) {  clearTimeout(method.tId)  method.tId = setTimeout(() => {    method.call(context)  }, 1000)}function print() {  console.log('Hello World')}window.onresize = debounce(print)复制代码

我们不停缩放窗口,当停止1S后,打印出Hello World。

有个可以优化的地方: 此实现方法有副作用(Side Effect),改变了输入值(method),给method新增了属性

2. 优化第一版:消除副作用,将定时器隔离

function debounce(method, wait, context) {  let timeout  return function() {    if (timeout) {      clearTimeout(timeout)    }    timeout = setTimeout(() => {      method.call(context)    }, wait)  }}复制代码

3. 优化第二版:自动调整this正确指向

之前的函数我们需要手动传入函数执行上下文context,现在优化将 this 指向正确的对象。

function debounce(method, wait) {  let timeout  return function() {    // 将method执行时this的指向设为debounce返回的函数被调用时的this指向    let context = this    if (timeout) {      clearTimeout(timeout)    }    timeout = setTimeout(() => {      method.call(context)    }, wait)  }}复制代码

4. 优化第三版:函数可传入参数

即便我们的函数不需要传参,但是别忘了JavaScript 在事件处理函数中会提供事件对象 event,所以我们要实现传参功能。

function debounce(method, wait) {  let timeout  // args为返回函数调用时传入的参数,传给method  return function(...args) {    let context = this    if (timeout) {      clearTimeout(timeout)    }    timeout = setTimeout(() => {      // args是一个数组,所以使用fn.apply      // 也可写作method.call(context, ...args)      method.apply(context, args)    }, wait)  }}复制代码

5. 优化第四版:提供立即执行选项

有些时候我不希望非要等到事件停止触发后才执行,我希望立刻执行函数,然后等到停止触发n毫秒后,才可以重新触发执行。

function debounce(method, wait, immediate) {  let timeout  return function(...args) {    let context = this    if (timeout) {      clearTimeout(timeout)    }    // 立即执行需要两个条件,一是immediate为true,二是timeout未被赋值或被置为null    if (immediate) {      // 如果定时器不存在,则立即执行,并设置一个定时器,wait毫秒后将定时器置为null      // 这样确保立即执行后wait毫秒内不会被再次触发      let callNow = !timeout      timeout = setTimeout(() => {        timeout = null      }, wait)      if (callNow) {        method.apply(context, args)      }    } else {      // 如果immediate为false,则函数wait毫秒后执行      timeout = setTimeout(() => {        // args是一个类数组对象,所以使用fn.apply        // 也可写作method.call(context, ...args)        method.apply(context, args)      }, wait)    }  }}复制代码

6. 优化第五版:提供取消功能

有些时候我们需要在不可触发的这段时间内能够手动取消防抖,代码实现如下:

function debounce(method, wait, immediate) {  let timeout  // 将返回的匿名函数赋值给debounced,以便在其上添加取消方法  let debounced = function(...args) {    let context = this    if (timeout) {      clearTimeout(timeout)    }    if (immediate) {      let callNow = !timeout      timeout = setTimeout(() => {        timeout = null      }, wait)      if (callNow) {        method.apply(context, args)      }    } else {      timeout = setTimeout(() => {        method.apply(context, args)      }, wait)    }  }  // 加入取消功能,使用方法如下  // let myFn = debounce(otherFn)  // myFn.cancel()  debounced.cancel = function() {    clearTimeout(timeout)    timeout = null  }}复制代码

至此,我们已经比较完整地实现了一个underscore中的debounce函数。

六、遗留问题

需要防抖的函数可能是存在返回值的,我们要对这种情况进行处理,underscore的处理方法是将函数返回值在返回的debounced函数内再次返回,但是这样其实是有问题的。如果参数immediate传入值不为true的话,当防抖后的函数第一次被触发时,如果原始函数有返回值,其实是拿不到返回值的,因为原函数是在setTimeout内,是异步延迟执行的,而return是同步执行的,所以返回值是undefined

第二次触发时拿到的返回值其实是第一次执行的返回值,第三次触发时拿到的返回值其实是第二次执行的返回值,以此类推。

1. 使用回调函数处理函数返回值

function debounce(method, wait, immediate, callback) {  let timeout, result  let debounced = function(...args) {    let context = this    if (timeout) {      clearTimeout(timeout)    }    if (immediate) {      let callNow = !timeout      timeout = setTimeout(() => {        timeout = null      }, wait)      if (callNow) {        result = method.apply(context, args)        // 使用回调函数处理函数返回值        callback && callback(result)      }    } else {      timeout = setTimeout(() => {        result = method.apply(context, args)        // 使用回调函数处理函数返回值        callback && callback(result)      }, wait)    }  }  debounced.cancel = function() {    clearTimeout(timeout)    timeout = null  }  return debounced}复制代码

这样我们就可以在函数防抖时传入一个回调函数来处理函数的返回值,使用代码如下:

function square(num) {  return Math.pow(num, 2)}let debouncedFn = debounce(square, 1000, false, val => {  console.log(`原函数的返回值为:${val}`)})window.addEventListener('resize', () => {  debouncedFn(4)}, false)// 停止缩放1S后输出:// 原函数的返回值为:16复制代码

2. 使用Promise处理返回值

function debounce(method, wait, immediate) {  let timeout, result  let debounced = function(...args) {    // 返回一个Promise,以便可以使用then或者Async/Await语法拿到原函数返回值    return new Promise(resolve => {      let context = this      if (timeout) {        clearTimeout(timeout)      }      if (immediate) {        let callNow = !timeout        timeout = setTimeout(() => {          timeout = null        }, wait)        if (callNow) {          result = method.apply(context, args)          // 将原函数的返回值传给resolve          resolve(result)        }      } else {        timeout = setTimeout(() => {          result = method.apply(context, args)          // 将原函数的返回值传给resolve          resolve(result)        }, wait)      }    })  }  debounced.cancel = function() {    clearTimeout(timeout)    timeout = null  }  return debounced}复制代码

使用方法一:在调用防抖后的函数时,使用then拿到原函数的返回值

function square(num) {  return Math.pow(num, 2)}let debouncedFn = debounce(square, 1000, false)window.addEventListener('resize', () => {  debouncedFn(4).then(val => {    console.log(`原函数的返回值为:${val}`)  })}, false)// 停止缩放1S后输出:// 原函数的返回值为:16复制代码

使用方法二:调用防抖后的函数的外层函数使用Async/Await语法等待执行结果返回

使用方法见代码:

function square(num) {  return Math.pow(num, 2)}let debouncedFn = debounce(square, 1000, false)window.addEventListener('resize', async () => {  let val  try {    val = await debouncedFn(4)  } catch (err) {    console.error(err)  }  console.log(`原函数返回值为${val}`)}, false)// 停止缩放1S后输出:// 原函数的返回值为:16复制代码

七、参考文章

如果有错误或者不严谨的地方,请务必给予指正,十分感谢。

转载地址:http://bmlpa.baihongyu.com/

你可能感兴趣的文章
Eclipse Open J9:Eclipse OMR项目提供的开源JVM
查看>>
HTTP内容分发——《HTTP权威指南》系列
查看>>
PHP autoload 机制详解
查看>>
302. Smallest Rectangle Enclosing Black Pixels
查看>>
从面向服务架构(SOA)学习:微服务时代应该借鉴的5条经验教训
查看>>
PostgreSQL Role Management
查看>>
如何使用 Swift 开发简单的条形码检测器?
查看>>
【腾讯Bugly干货分享】一步一步实现 Android 的 MVP 框架
查看>>
[设计模式]唠唠依赖注入
查看>>
Docker-Shiny使用笔记1
查看>>
[LintCode] Interval Minimum Number
查看>>
Stimulsoft Reports.Net开发者常见问题及解决方案汇总
查看>>
使用原生js+css3实现下拉刷新效果
查看>>
amazon access代码分析和记录
查看>>
SpringMVC无法访问根目录的问题
查看>>
使用 hooks 工程化 ionic 项目
查看>>
学习的艺术——如何学好一门技术、语言
查看>>
Javascript 中的 Array 操作
查看>>
前端之React实战-背景概述
查看>>
开坑,写点Polymer 1.0 教程第4篇——组件的生命周期
查看>>