

新闻资讯
行业动态防抖和节流是针对不同高频事件场景的互斥策略:防抖适用于“等用户彻底停手再执行”(如搜索、resize),节流适用于“必须定期响应但不能太密”(如拖拽、滚动);二者需根据是否需要最终状态或过程反馈来选择,且均非性能银弹,须先优化回调函数本身。
防抖和节流不是“选一个就行”的通用方案,而是针对不同高频事件场景的两种互斥策略:防抖适合“等用户彻底停手再执行”,节流适合“必须定期响应但不能太密”。
典型表现是:连续触发事件时,只在最后一次触发后延迟执行一次。
比如用户快速输入 5 个字符,debounce 会忽略前 4 次,只在停止输入 300ms 后调用一次搜索函数。
关键点:
setTimeout 必须在每次新触发时 clearTimeout,否则旧定时器仍会执行immediate 参数的版本可支持“先执行再等待”,但多数搜索场景不需要this 和参数,建议用 func.apply(context, args) 而非直接调用function debounce(func, wait, immediate = false) {
let timeout;
return function(...args) {
const later = () => {
timeout = null;
if (!immediate) func.apply(this, args);
};
const callNow = immediate && !timeout;
clearTimeout(timeout);
timeout = setTimeout(later, wait);
if (callNow) func.apply(this, args);
};
}典型表现是:无论事件触发多频繁,函数最多每隔 wait 毫秒执行一次。比如滚动事件每 16ms 触发一次,throttle 可将其限制为每 100ms 最多执行一次。
常见陷阱:
throttle 和 CSS 的 pointer-events: none 混用——后者阻断事件,前者只是降频requestAnimationFrame 替代function throttle(func, wait) {
let previous = 0;
return function(...args) {
const now = Date.now();
if (now - previous >= wait) {
func.apply(this, args);
previous = now;
}
};
}输入框搜索、表单校验、自动保存——这些操作只关心用户“最终输入了什么”,用 debounce 更合理;而 Canvas 绘图、滚动视差、游戏按键响应——这些依赖中间状态的场景,throttle 或 requestAnimationFrame 才合适。
性能影响差异:
debounce 在高频期完全不执行,内存占用低,但响应有延迟throttle 保持固定节奏,CPU 占用略高,但行为可预测useCallback(React)或手动缓存计算结果当目标是“每帧最多更新一次 UI”,比如监听 scroll 更新 sticky 导航栏位置,requestAnimationFrame 比 throttle(16) 更精准,且天然与浏览器刷新节奏同步。
注意点:
rafId 并在组件卸载/事件解绑时 cancelAnimationFrame
let rafId = null;
function animateScroll() {
// 更新 UI
rafId = requestAnimationFrame(animateScroll);
}
// 启动
rafId = requestAnimationFrame(animateScroll);
// 清理
if (rafId) cancelAnimationFrame(rafId);真正容易被忽略的是:防抖和节流本身不是性能银弹。如果回调函数里做了深克隆、正则全局匹配或未加索引的数组遍历,再怎么节流也救不了。先定位瓶颈,再决定用哪个“流控开关”。