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)
        }
    }
}
    
相关推荐
有意义11 分钟前
从 useState 到 useEffect:React Hooks 核心机制详解
javascript·react.js·前端工程化
栀秋66613 分钟前
面试常考的最长递增子序列(LIS),到底该怎么想、怎么写?
前端·javascript·算法
Zyx200718 分钟前
手写 `instanceof`:深入理解 JavaScript 原型链与继承机制
javascript
江城开朗的豌豆32 分钟前
TypeScript和JavaScript到底有什么区别?
前端·javascript
前端不太难2 小时前
如何给 RN 项目设计「不会失控」的导航分层模型
前端·javascript·架构
用户4099322502122 小时前
Vue3中v-show如何通过CSS修改display属性控制条件显示?与v-if的应用场景该如何区分?
前端·javascript·vue.js
Zyx20072 小时前
JavaScript 中 this 的设计哲学与运行机制
javascript
A24207349302 小时前
JavaScript图表制作:从入门到精通
开发语言·javascript·信息可视化
瘦的可以下饭了2 小时前
Day03-APIs
javascript
BD_Marathon2 小时前
Vue3_简介和快速体验
开发语言·javascript·ecmascript