JS 入门通关手册(38):防抖与节流 原理 + 手写 + 实战场景(面试必考)

摘要

本文详细讲解前端高频面试与实战核心知识点 ------ 防抖和节流。从概念、原理、区别入手,提供可直接用于生产环境的手写代码,并结合搜索框、滚动加载、窗口 resize、按钮点击等真实业务场景演示用法,帮助你彻底掌握性能优化必备技能。


一、为什么要使用防抖与节流?

在前端开发中,像 scrollresizeinputmousemoveclick 这类事件触发频率极高。如果每次触发都执行复杂逻辑(AJAX、DOM 操作、计算),会严重占用资源,导致页面卡顿、接口请求泛滥。

防抖与节流就是为了控制高频事件的执行次数,做性能优化。


二、防抖(Debounce)

1. 概念

触发高频事件后,延迟 n 秒再执行函数; 如果 n 秒内再次触发,则重新计时。

一句话记忆:**"等你停了,我再执行"**典型场景:搜索框输入联想。

2. 手写基础防抖

javascript

运行

javascript 复制代码
function debounce(fn, delay = 300) {
  let timer = null;
  return function (...args) {
    // 再次触发时清除上一次定时器
    if (timer) clearTimeout(timer);
    timer = setTimeout(() => {
      fn.apply(this, args);
    }, delay);
  };
}

3. 立即执行版本(可选)

有些场景希望第一次触发立即执行,后续频繁触发不执行:

javascript

运行

javascript 复制代码
function debounce(fn, delay = 300, immediate = false) {
  let timer = null;
  return function (...args) {
    if (timer) clearTimeout(timer);

    // 立即执行
    if (immediate && !timer) {
      fn.apply(this, args);
    }

    timer = setTimeout(() => {
      fn.apply(this, args);
    }, delay);
  };
}

4. 防抖适用场景

  • 搜索框输入联想(input)
  • 表单验证(手机号、邮箱校验)
  • 窗口大小变化 resize
  • 鼠标移动 mousemove
  • 频繁点击按钮防止重复请求

三、节流(Throttle)

1. 概念

高频事件触发时,保证在 n 秒内只执行一次 **。无论触发多快,都按固定频率执行。**

一句话记忆:**"冷却时间,到点才能放技能"**典型场景:滚动加载、下拉刷新。

2. 时间戳版节流(立即执行)

javascript

运行

javascript 复制代码
function throttle(fn, interval = 300) {
  let lastTime = 0;
  return function (...args) {
    const now = Date.now();
    if (now - lastTime >= interval) {
      fn.apply(this, args);
      lastTime = now;
    }
  };
}

3. 定时器版节流(延迟执行)

javascript

运行

javascript 复制代码
function throttle(fn, interval = 300) {
  let timer = null;
  return function (...args) {
    if (!timer) {
      timer = setTimeout(() => {
        fn.apply(this, args);
        timer = null;
      }, interval);
    }
  };
}

4. 节流适用场景

  • 滚动加载更多(scroll)
  • 高频点击抢购、投票
  • 鼠标拖拽、绘图
  • 游戏技能冷却
  • 视频播放时间更新

四、防抖 vs 节流 核心区别

表格

特性 防抖 Debounce 节流 Throttle
核心思想 最后一次才执行 固定频率执行
执行次数 多次触发 → 一次执行 多次触发 → 均匀执行
口诀 停了再执行 冷却再执行
典型场景 搜索输入、表单校验 滚动加载、点击限流

一句话区分:

  • 防抖:只认最后一次
  • 节流:均匀执行

五、真实业务使用示例

1. 搜索框防抖

javascript

运行

javascript 复制代码
const input = document.querySelector("#search");
input.oninput = debounce(() => {
  console.log("发起搜索请求");
}, 500);

2. 滚动节流加载

javascript

运行

javascript 复制代码
window.onscroll = throttle(() => {
  console.log("滚动加载数据");
}, 800);

六、高频面试题

  1. **防抖和节流的区别是什么?**答:防抖是多次触发只执行最后一次;节流是在固定时间内只执行一次,均匀限流。

  2. **分别适用什么场景?**答:防抖用于输入、resize;节流用于滚动、点击、拖拽。

  3. **手写防抖 / 节流要注意什么?**答:保存定时器 / 时间戳、返回闭包、绑定 this、传递参数。

  4. **节流时间戳和定时器版本区别?**答:时间戳版立即执行,定时器版延迟执行。

  5. **Vue/React 中使用要注意什么?**答:注意防抖节流函数只初始化一次,不要放在 render 内重复生成。


七、工程实践建议

  1. 搜索输入、表单校验 → 防抖
  2. 滚动加载、高频点击 → 节流
  3. 延迟优先 300~500ms
  4. 项目中可直接使用 Lodash:_.debounce / _.throttle
  5. 面试必须能手写最简版本

七、总结

  1. 防抖:高频触发 → 等待停止 → 执行最后一次
  2. 节流:高频触发 → 固定频率 → 均匀执行
  3. 防抖适合 "停止后执行",节流适合 "持续触发限流"
  4. 两者都是前端性能优化必备手段
  5. 手写核心:闭包保存定时器 / 时间戳,控制执行时机
相关推荐
云水一下2 小时前
TypeScript 从零基础到精通(五):高级类型与泛型
前端·javascript·typescript
云水一下2 小时前
TypeScript 从零基础到精通(六):类型声明与模块化
javascript·typescript
xiaofeichaichai3 小时前
Map / Set / WeakMap / WeakSet
前端·javascript
有梦想的程序星空5 小时前
【环境配置】Vue3项目离线化本地部署echarts全攻略
前端·javascript·vue·echarts
薛先生_0996 小时前
vue-路由重定向
前端·javascript·vue.js
橘子星6 小时前
基于 ES6 语法的 NLP 任务模块化开发实践
前端·javascript
月光刺眼6 小时前
JS 底层执行机制探讨:执行上下文、变量提升与调用栈
前端·javascript
ZC跨境爬虫7 小时前
跟着 MDN 学 JavaScript day_1:什么是 JavaScript?
开发语言·前端·javascript·ecmascript
xiaofeichaichai7 小时前
Vue 响应式原理
前端·javascript·vue.js
提子拌饭1337 小时前
模态窗鸿蒙PC Electron框架实现技术详解 - 饮料含糖量应用案例分析
前端·javascript·华为·electron·前端框架·开源·鸿蒙