前端总说的防抖与节流到底是什么?

一、 一句话说明白

  • 防抖 是"等一等",等到最后一次操作才执行。
  • 节流 是"省一省",不管多快,都按固定频率执行。

二、 防抖 (Debounce)

防抖的原理是:在事件被触发 n 秒后再执行回调,如果在这 n 秒内又被触发,则重新计时。

可以将防抖理解为"最后一次说了算"。

其核心在于延迟执行,只有当用户停止操作一段时间后,才真正执行逻辑。

以下是一个具备通用性的防抖函数实现,利用闭包保存定时器状态。

JavaScript 复制代码
/**
 * 防抖函数 (Debounce)
 * @param {Function} fn - 需要执行的函数
 * @param {Number} delay - 延迟时间 (毫秒)
 * @returns {Function} - 包装后的函数
 */
function debounce(fn, delay) {
  let timer = null;

  return function (...args) {
    // 如果此时再次触发,则清除上一次的定时器,重新计时
    if (timer) clearTimeout(timer);

    // 保存当前的 this 上下文,确保 fn 执行时 this 指向正确
    const context = this;

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

// --- 使用示例 ---
const handleInput = debounce((e) => {
  console.log('搜索请求发送,关键词:', e.target.value);
}, 500);

// 绑定到输入框
// document.getElementById('search').addEventListener('input', handleInput);
  • 搜索框输入联想 (Input Search): 用户在不断输入字符时,不应立即请求后端接口,而是等待用户停止输入(如停顿 300ms)后再发送请求。
  • 窗口大小调整 (Window Resize): 只需要在窗口调整结束后计算一次布局,而非调整过程中不断计算。
  • 表单验证: 避免每输入一个字符就验证一次,而是输入完成后验证。

三、 节流 (Throttle)

节流的原理是:规定在一个单位时间内,只能触发一次函数。如果这个单位时间内触发多次函数,只有一次生效。

可以将节流理解为"按照固定的频率执行"。

无论事件触发得多么频繁,执行频率都是稀疏且固定的。

节流通常有两种实现方式:时间戳版 (立即执行)和定时器版(延后执行)。

以下推荐一种基于时间戳的高效实现,适合滚动等需要立即响应的场景。

JavaScript 复制代码
/**
 * 节流函数 (Throttle) - 时间戳版
 * @param {Function} fn - 需要执行的函数
 * @param {Number} interval - 间隔时间 (毫秒)
 * @returns {Function} - 包装后的函数
 */
function throttle(fn, interval) {
  let lastTime = 0;

  return function (...args) {
    const now = Date.now();
    
    // 如果当前时间与上次执行时间的差值大于设定间隔,则执行
    if (now - lastTime >= interval) {
      fn.apply(this, args); // 修正 this 指向并传递参数
      lastTime = now;
    }
  };
}

// --- 使用示例 ---
const handleScroll = throttle(() => {
  console.log('触发滚动检测', window.scrollY);
}, 200);

// 绑定到滚动事件
// window.addEventListener('scroll', handleScroll);
  • 页面滚动 (Scroll): 监听滚动条位置以实现"懒加载"或"吸顶"效果。不需要每像素移动都执行,每隔 100ms 检查一次即可。
  • DOM 元素拖拽 (Drag and Drop): 在拖拽过程中通过节流限制计算位置的频率,避免卡顿。
  • 高频点击提交: 防止用户疯狂点击按钮导致多次提交表单。

四、成熟的工具库

在实际的大型项目中,除非为了减少依赖体积或特定需求,通常不建议手写上述基础函数,因为需要处理更复杂的边界情况(如取消操作、立即执行选项等)。

  1. Lodash: 使用 _.debounce_.throttle

    • 例如:_.debounce(func, [wait=0], [options={}]) 支持 leading (前沿触发) 和 trailing (后沿触发) 配置。
  2. RequestAnimationFrame: 对于涉及动画或极高频的 UI 渲染(如拖拽),现代浏览器推荐使用 requestAnimationFrame 代替传统的定时器节流,以获得更流畅的视觉效果 (60FPS)。

相关推荐
zzjyr1 小时前
react前端项目 fetch原生 与 umijs request 四种请求区别
前端
小时前端1 小时前
微信小程序选不了本地文件?用 web-view + H5 一招搞定
前端·微信小程序·uni-app
71Ove1 小时前
告别手写字符串!UniApp 路由全自动类型生成工具
前端
掘金安东尼2 小时前
从平面到空间:用 React Three Fiber 构建 3D 产品网格
前端·javascript·面试
小时前端2 小时前
HTTPS 页面加载 HTTP 脚本被拦?同源代理来救场
前端·https
用户683709359552 小时前
在 Rokid AR 眼镜里玩消消乐:基于 Unity 2022 LTS + UXR 3.0 SDK 的轻量级 AR 游戏尝试
前端
zzjyr2 小时前
@umijs/max 中导出的 request 方法,如何实现 GET/POST/PUT/DELETE 这四种核心请求
前端
swipe2 小时前
#用这 9 个浏览器 API,我把页面从“卡成 PPT”救回到 90+(每个都有能直接抄的例子)
前端·javascript·面试
zzjyr2 小时前
基于 @umijs/max 的 request 补充常见错误统一处理、请求取消、重复请求防抖的完整方案
前端