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. 手写核心:闭包保存定时器 / 时间戳,控制执行时机
相关推荐
A_nanda8 分钟前
VS2022安装QT6.5.3后,如何更新项目配置
前端·javascript·vue.js
heyCHEEMS33 分钟前
记录一下自动化构建中 SSE 与子进程管理的三个坑
javascript·node.js
SonoTommy33 分钟前
在 Node.js 文件上传中集成 ClamAV 扫描
javascript
Live&&learn2 小时前
Vue项目打包后内联字符串不显示的原因
前端·javascript·vue.js
ejinxian3 小时前
Rust的GUI方案中,Slint、Azul、egui、iced、Druid、Tauri
前端·javascript·vue.js
Python私教3 小时前
ShadcnVueAdmin 的国际化是怎么实现的
前端·javascript·vue.js
㳺三才人子3 小时前
容器內的 H2 控制台
开发语言·前端·javascript
薛定谔的猫19824 小时前
gradio学习代码部分
java·前端·javascript
yqcoder4 小时前
React 深度解析:类组件 (Class) vs 函数组件 (Function)
前端·javascript·react.js
HyaCinth4 小时前
一人一周,用 Codex 渐进式迁移重构了一个材料学组件库
前端·javascript·css