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秒内触发多次函数,只有一次生效,节流会稀释函数的执行频率。

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

学到知识了就点个赞叭

相关推荐
天若有情6731 天前
程序员原创|借鉴JS事件冒泡,根治电脑文件混乱的“冒泡整理法”
开发语言·javascript·windows·ecmascript·电脑·办公·日常
FYKJ_20101 天前
springboot校园兼职平台--附源码02041
java·javascript·spring boot·python·eclipse·django·php
空中海1 天前
01 React Native 基础、核心组件与布局体系
javascript·react native·react.js
前端之虎陈随易1 天前
2年没用Nodejs了,Bun很香
linux·前端·javascript·vue.js·typescript
好运的阿财1 天前
OpenClaw工具拆解之host_workspace_write+host_workspace_edit
前端·javascript·人工智能·机器学习·ai编程·openclaw·openclaw工具
XiYang-DING1 天前
JavaScript
开发语言·javascript·ecmascript
空中海1 天前
02 React Native状态、导航、数据流与设备能力
javascript·react native·react.js
空中海2 天前
02 状态、Hooks、副作用与数据流
开发语言·javascript·ecmascript
空中海2 天前
04 React Native工程化、质量、发布与生态选型
javascript·react native·react.js