js中,防抖、节流的原理和区别

在前端开发的过程中,会有很多场景会频繁触发事件,比如说搜索框实时发请求,onmousemove, resize, onscroll等等,但有些时候我们并不希望在事件持续触发的过程中那么频繁地去执行函数。,咋办呢?这时候就应该用到函数防抖和函数节流了!

为什么要防抖

有的操作是高频触发的,但是其实触发一次就好了,比如我们短时间内多次缩放页面,不应该每次缩放都去执行操作,应该只做一次就好。再比如说监听输入框的输入,不应该每次都触发监听,应该是用户完成一段驶入后再进行触发
防抖就是防止抖动,避免事件的重复触发

思路:事件触发后开启一个定时器,如果事件在这个定时器限定的时间内再次触发,则清除定时器,再写一个定时器,定时器时间到则触发

举个栗子 复制代码
 // 1.声明定时器变量
    // 2.每次鼠标移动的时候要先判断是否有定时器  如果有先清除以前的定时器
    // 3.如果没有定时器  执行定时器  存储变量
    // 4.定时器里写函数
    const obox = document.querySelector('div')
    let i = 1;
    function mousemove() {
      obox.innerHTML = i++;
    }
    function debounce(fun, time) {
      let timer
      return function () {
        let that = this;
        if (timer) clearTimeout(timer)
        timer = setTimeout(function () {
          fun.call(that)
        }, time)
      }
    }
    obox.addEventListener('mousemove', debounce(mousemove, 1000))

为什么要节流

防抖存在一个问题,事件会一直等到用户完成操作后一段时间再操作,如果一直操作,会一直不触发。比如说是一个按钮,点击就发送请求,如果一直点,那么请求就会一直发布出去。这里正确的思路应该是第一次点击就发送,然后上一个请求回来后,才能再发
节流就是减少流量,将频繁触发的事件减少,并每隔一段一段时间执行。即控制事件触发的频率

两种节流实现方式

  • 使用定时器实现
javascript 复制代码
function fn(fun,t){
      let timer = null;
      return function(){
        let that = this;
        if(!timer){
          timer = setTimeout(function(){
            fun.call(that)
            timer = null
          },t)
        }
      }
    }
    obox.addEventListener('mousemove',fn(mousemove,1000))

上面代码中,触发事件后,会直接重新赋值timer为一个定时器,到时间后执行事件处理程序,重新赋值timer为null。在这期间,因为timer的值是一个定时器,if判断是不会执行的,必须重新赋值为null后才会重新执行

  • 时间戳实现
kotlin 复制代码
 function fn(fun, time) {
      let start = 0;
      return function () {
        let that = this;
        let current = Date.now()
        if (current - start >= time) {
          fun.call(that)
          start = current
        }
      }
    }
    obox.addEventListener('mousemove', fn(mousemove, 1000))

上面代码中,触发事件后,定义开始触发事件的时间为start为零,获取当前的时间戳赋值给current,判断当前时间与初始时间是否超过间隔,如果超过了就把当前这个时间戳赋值给start,然后执行事件处理程序。(因为初始时间为零,所以第一次触发会直接执行)

总结

  • 防抖和节流相同点:

防抖和节流都是为了阻止操作高频触发,从而浪费性能。

防抖和节流区别:

防抖是触发高频事件后n秒内函数只会执行一次,如果n秒内高频事件再次被触发,则重新计算时间。适用于可以多次触发但触发只生效最后一次的场景。
节流是高频事件触发,但在n秒内只会执行一次,如果n秒内触发多次函数,只有一次生效,节流会稀释函数的执行频率。

今天的知识就分享到这里啦。欢迎大家在评论里指点讨论

学到知识了就点个赞叭

相关推荐
英俊潇洒美少年5 小时前
Vue 生产环境打包:SourceMap、压缩、混淆、加密全解 + 最佳实践
前端·javascript·vue.js
巴博尔5 小时前
UNIAPP中NVUE页面 动画
android·前端·javascript·ios·uni-app
猫头虎-前端技术6 小时前
JS 作用域与闭包:从变量提升到闭包陷阱的超详细解析
开发语言·javascript·云计算·bootstrap·ecmascript·openstack·perl
她说人狗殊途7 小时前
基于 vue-cli 创建
前端·javascript·vue.js
大家的林语冰8 小时前
Deno 2.8 正式发布,再次超越 Bun,史上最大的次版本升级诞生!
前端·javascript·node.js
影寂ldy9 小时前
C#数组的属性和方法(Clear / Copy / IndexOf )
开发语言·javascript·c#
Brave & Real9 小时前
小程序 const 在js中以及与同类的var和let之间的差异
javascript·微信小程序·小程序
米丘10 小时前
React 19.x 的 lazy 与 Suspense
前端·javascript·react.js
ZC跨境爬虫10 小时前
跟着 MDN 学CSS day_21:(图像溢出控制与表单元素样式定制)
前端·javascript·css·ui·交互
riuphan11 小时前
JavaScript 中的 this 关键字
javascript