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)
        }
    }
}
    
相关推荐
帆张芳显3 小时前
智表 ZCELL 公式引擎,帮你解锁自定义函数与跨表计算的强大能力
前端·javascript
渣哥4 小时前
Lazy能否有效解决循环依赖?答案比你想的复杂
javascript·后端·面试
北城以北88884 小时前
Vue-- Axios 交互(一)
前端·javascript·vue.js
gplitems1234 小时前
Spide - Personal Blog & Magazine WordPress Theme Download
javascript
前端开发爱好者4 小时前
Vite➕ 收费了!
前端·javascript·vue.js
细节控菜鸡5 小时前
【2025最新】ArcGIS for JS 范围裁剪(只保留特定区域显示),实现精准地理范围聚焦
开发语言·javascript·arcgis
前端小菜袅5 小时前
uniapp配置自动导入uni生命周期等方法
前端·javascript·uni-app
一枚前端小能手5 小时前
「周更第7期」实用JS库推荐:Vite
前端·javascript·vite
知识分享小能手6 小时前
微信小程序入门学习教程,从入门到精通,电影之家小程序项目知识点详解 (17)
前端·javascript·学习·微信小程序·小程序·前端框架·vue