JS:手搓一份节流函数

前端开发中,我们经常会遇到「高频触发事件」的场景 ------ 比如用户快速滑动页面、频繁点击按钮、实时输入搜索框等。如果直接在这些事件的回调函数中执行复杂逻辑(如 DOM 操作、接口请求),会导致浏览器频繁计算渲染,进而出现页面卡顿、响应延迟等问题。而「节流(Throttle)」作为前端性能优化的核心技术之一,能有效解决这类问题,它就像一个 "流量闸门",控制函数在指定时间内只执行一次,避免资源被过度消耗。

  1. 节流的定义

节流的核心逻辑可以概括为:在连续触发的事件中,确保函数在固定时间间隔内只执行一次。无论事件触发多少次(比如 1 秒内触发 100 次),函数最多只会执行 1 次,剩余的触发会被 "拦截",直到下一个时间间隔开启。​

举个生活中的例子:淋浴时调节水龙头的 "节水模式"------ 即使一直按住开关,水流也会按照固定频率间断流出,而不是持续喷涌。节流对函数的控制,就像这个 "节水模式",控制函数的 "执行频率"。

利用时间戳实现一个最后一次不执行的节流函数

梳理雏形

js 复制代码
    //节流函数和防抖函数都是传入,一个函数,一个时间,最后返回一个函数
    const throttle = (fn, delay) => {
    
    return function () {
            fn()
        }
}
    

核心逻辑: 通过记录「上次函数执行的时间戳」,每次事件触发时,对比当前时间与上次执行时间的差值,利用闭包机制保存上次的time:​

  • 若差值大于指定间隔 (delay),则立即执行函数,并更新上次执行时间;
  • 若差值小于间隔 ,则跳过本次触发。
js 复制代码
const throttle = (fn, delay) => {
    let time = 0;
    return function () {
        const date = Date.now()
        //初始时间 date-time> delay 恒成立 因为1970年1月1日 是时间戳 0 的时间
        if (date - time > delay) {
            time = date
            fn()
        }
    }
}

处理 this 指向和收集传入的参数就得到了,最后一次不执行的节流函数

js 复制代码
const throttle = (fn, delay) => {
    let time = 0;
    return function (...args) {
        const date = Date.now()
        if (date - time > delay) {
            time = date
            fn.apply(this, args)
        }
    }
}

如果业务中需要持续的数据处理,需要触发最后一次的函数,则需要加上定时器

  • 首次触发时,若没有定时器,则设置一个定时器,在间隔后执行函数;​

  • 连续触发时,若已存在定时器,则跳过本次触发;​

  • 定时器执行时,更新上次执行时间,并清空定时器,确保下次触发可正常执行。

沿用时间戳 节流的代码,增加变量记录剩余时间

js 复制代码
const throttle = (fn, delay) => {
    let timer = null;
    let time = 0;
    return function (...args) {
        if (timer) {
            clearTimeout(timer)
            //保证每一次执行的时候把之前预执行的函数取消
        }
        const now = Date.now()
        const remainTime = now - time - delay
        if (remainTime < 0) {
            timer = setTimeout(() => {
                time = Date.now()//记录当前执行的时间
                fn.apply(this, ...args)
            }, remainTime)
            //经过剩余时间后执行函数,完成最后一次的交互
        } else {
            time = Date.now()//记录当前执行的时间
            fn.apply(this, ...args)
        }
    }
}
    
相关推荐
竹林8184 小时前
用 wagmi v2 + viem 监听链上事件,我踩了三天坑终于搞懂了实时日志与历史补全
javascript
只一4 小时前
😭从回调地狱到 async/await:一文打通 Ajax 与 JS 异步编程
javascript
weedsfly4 小时前
语法糖褪去之后——Babel 转译产物中的 JavaScript 本貌
前端·javascript
JustHappy4 小时前
「软件设计思想杂谈🤔」“切图仔”也能懂编译原理?框架源码也许没那么难。聊聊 Vue 的编译(上)
前端·javascript·vue.js
晓得迷路了5 小时前
栗子前端技术周刊第 134 期 - React Router v8、TypeScript 7 RC、React Native 0.86...
前端·javascript·react.js
代码煮茶21 小时前
React 组件封装方法论 —— 以 Todo App 为例
javascript·react.js
任沫21 小时前
Agent之Function Call
javascript·人工智能·go
默_笙1 天前
🛬 我让 AI 帮我写了一个打飞机游戏,结果 Canvas 把我整不会了
前端·javascript
梯度不陡1 天前
AI 到底能不能从零写软件?ProgramBench 和 RepoZero 给出了两种答案
前端·javascript·面试
胡萝卜术1 天前
滑动窗口最大值:从暴力到单调队列,层层优化全解析
前端·javascript·面试