防抖函数中的this和闭包

前言

最近我们学习了闭包,this,对象,作用域的一些基本知识,今天蘑菇头就来分享一个案例,将这些知识都串联起来。一起来看看你还记得多少。

防抖

今天蘑菇头要聊的案例叫做防抖,那么什么是防抖呢?在前端开发中,防抖(Debounce)是一种常用的优化技术。它主要用于处理频繁触发的事件(如滚动、输入等)。其基本原理是,在事件触发后,延迟一段时间再执行相关的操作,如果在这段延迟时间内又有新的事件触发,则重新开始计时,直到延迟时间结束且没有新的事件触发才真正执行操作。这样可以避免因为频繁触发事件而导致不必要的大量计算或请求,提高性能和用户体验。我们用一个小demo来实现一下防抖。

案例

我们写一个按钮,当点击这个按钮一下就打印一句话用来模拟发请求这个动作。

当我们一直点击提交这个按钮时,他会一直发送请求到我们的后台服务器,这些请求都是同一个请求,后台服务器有可能会因为承受不住这样的压力而奔溃,所以我们需要设计一个方法,当用户一直连续点击提交时,不发送请求到后端,而是停止点击后发送请求,这样就能给我们后台服务器减轻压力,实现优化。

思路历程

  1. 首先我们获取到提交按钮,给他添加点击事件监听,当点击时,发送请求。
javascript 复制代码
let btn = document.getElementById('btn');
btn.addEventListener('click', function(){
     handle();
});         
//模拟请求
function handle() {
      console.log('人生无常,大肠包小肠');
}
  1. 我们知道有一个函数叫做setTimeout,他的作用是当过完设定好的时间之后执行回调函数。那么我们可以试着将请求放进回调函数里,看看能不能行。
javascript 复制代码
btn.addEventListener('click', function(){
     setTimeout(function(){
         handle();
     },1000)
}); 

可以发现,当我们一直连续点击时,他会等一段时间后疯狂打印,这显然不是我们想要的效果,我们想要的是当一直连续点击后发送一个请求,也就是打印一句话。

  1. 继续优化,设计一个防抖函数出来
javascript 复制代码
btn.addEventListener('click', debence(handle));
//模拟请求
function handle() {
      console.log('人生无常,大肠包小肠');
}
//防抖函数
function debence(handle){
    let timer = null;
    return function (e) {
    //如果第二次的时间没到1秒,就销毁上一个定时器
    // 第二次点击时,会找到上一次闭包中的timer,并且销毁
    clearTimeout(timer)
    timer = setTimeout(()=>{
          handle.call(this,e);
         }, 1000);
    }
}

我们利用闭包的原理,内部函数一定可以访问外部函数的变量,当内部函数被拿到外部函数之外调用时,即使外部函数执行完毕,但是内部函数对外部函数中的变量依然存在引用,那么这些被引用的变量会以一个集合的方式保存下来。我们设计一个timer将定时器保存下来,当第二次点击在规定的时间内时,由于第一次点击的事件还未结束,并且timer被闭包保存下来了,所以第二次点击时他会去找闭包中的timer销毁,这样第一次点击创建的定时器以及里面的请求就会被销毁,然后会创建一个新的timer,这样定时器倒计时间就完成更新了。

问题

  1. this指向问题

    注意,此时的handle的this是指向全局的,并不是指向dom结构的,所以我们需要更改this的指向,使用显示绑定规则,handle.call(this),这个this是谁有必要说明一下,箭头函数没有this,所以去上一级找,不是定时器的this,而是这个return函数的this,也就是debence(handle)的this,它是指向dom结构的,我们需要同步handle的this到dom结构上。

  2. 事件参数e的问题

    我们需要将事件参数e给到handle函数,所以还需在return函数传参,并且给到handle,这样我们的handle才能拿到事件参数。

总结

今天我们学习了什么是防抖以及如何手搓一个简单的防抖函数,并且里面还有一些注意事项需要了解。

相关推荐
桃园码工3 分钟前
15_HTML5 表单属性 --[HTML5 API 学习之旅]
前端·html5·表单属性
百万蹄蹄向前冲44 分钟前
2024不一样的VUE3期末考查
前端·javascript·程序员
轻口味1 小时前
【每日学点鸿蒙知识】AVCodec、SmartPerf工具、web组件加载、监听键盘的显示隐藏、Asset Store Kit
前端·华为·harmonyos
alikami1 小时前
【若依】用 post 请求传 json 格式的数据下载文件
前端·javascript·json
wakangda2 小时前
React Native 集成原生Android功能
javascript·react native·react.js
吃杠碰小鸡2 小时前
lodash常用函数
前端·javascript
emoji1111112 小时前
前端对页面数据进行缓存
开发语言·前端·javascript
泰伦闲鱼2 小时前
nestjs:GET REQUEST 缓存问题
服务器·前端·缓存·node.js·nestjs
m0_748250032 小时前
Web 第一次作业 初探html 使用VSCode工具开发
前端·html
一个处女座的程序猿O(∩_∩)O2 小时前
vue3 如何使用 mounted
前端·javascript·vue.js