0

0

js如何实现防抖函数 防抖函数的3种经典实现方案

尼克

尼克

发布时间:2025-06-23 21:52:02

|

1020人浏览过

|

来源于php中文网

原创

防抖函数的核心是延迟执行并重置计时,适用于搜索建议等场景。具体实现通过settimeout延迟触发函数,若再次触发则清除原定时器并重新计时。对于需要立即执行的情况,可引入isinvoked变量控制首次执行。同时可通过添加cancel方法实现手动取消。使用apply确保上下文和参数正确传递。防抖与节流的区别在于防抖保证最后一次触发有效,而节流确保一段时间内仅执行一次。常见应用场景包括搜索建议、窗口调整、按钮防重复点击、文本自动保存等。

js如何实现防抖函数 防抖函数的3种经典实现方案

防抖函数的核心在于:在事件触发后,不是立即执行,而是等待一段时间,如果在这段时间内再次触发,就重新计时。最终只执行最后一次触发。

js如何实现防抖函数 防抖函数的3种经典实现方案

解决方案

防抖的本质是延迟执行,并且在延迟的过程中,如果再次触发,则重新计时。理解这一点,实现防抖就变得简单了。

js如何实现防抖函数 防抖函数的3种经典实现方案
function debounce(func, delay) {
  let timeoutId;

  return function(...args) {
    const context = this;

    clearTimeout(timeoutId);

    timeoutId = setTimeout(() => {
      func.apply(context, args);
    }, delay);
  };
}

// 示例
function sayHello(name) {
  console.log(`Hello, ${name}!`);
}

const debouncedHello = debounce(sayHello, 300);

// 模拟事件触发
debouncedHello("Alice");
debouncedHello("Bob");
setTimeout(() => debouncedHello("Charlie"), 200); // 会取消Bob的执行
setTimeout(() => debouncedHello("David"), 500); // 延迟500ms,会执行

如何使用立即执行的防抖函数?

有时候,我们希望第一次触发事件时立即执行,而不是等待。可以修改防抖函数来实现:

js如何实现防抖函数 防抖函数的3种经典实现方案
function debounceImmediate(func, delay) {
  let timeoutId;
  let isInvoked = false;

  return function(...args) {
    const context = this;

    if (!isInvoked) {
      func.apply(context, args);
      isInvoked = true;
    }

    clearTimeout(timeoutId);

    timeoutId = setTimeout(() => {
      isInvoked = false;
    }, delay);
  };
}

// 示例
function sayHelloImmediate(name) {
  console.log(`Hello (Immediate), ${name}!`);
}

const debouncedHelloImmediate = debounceImmediate(sayHelloImmediate, 300);

debouncedHelloImmediate("Eve"); // 立即执行
debouncedHelloImmediate("Frank"); // 不会立即执行,但会重置计时
setTimeout(() => debouncedHelloImmediate("Grace"), 500); // 延迟500ms后执行

关键在于引入一个isInvoked变量,控制首次是否立即执行。这种方式适用于需要立即响应的场景,例如搜索框输入。

如何取消防抖函数的执行?

某些情况下,我们需要手动取消防抖函数的执行。可以给防抖函数添加一个cancel方法:

CopyWeb
CopyWeb

AI网页设计转换工具,可以将屏幕截图、网站URL转换为代码组件

下载
function debounceWithCancel(func, delay) {
  let timeoutId;

  function debounced(...args) {
    const context = this;

    clearTimeout(timeoutId);

    timeoutId = setTimeout(() => {
      func.apply(context, args);
    }, delay);
  }

  debounced.cancel = () => {
    clearTimeout(timeoutId);
  };

  return debounced;
}

// 示例
function sayHelloCancelable(name) {
  console.log(`Hello (Cancelable), ${name}!`);
}

const debouncedHelloCancelable = debounceWithCancel(sayHelloCancelable, 300);

debouncedHelloCancelable("Henry");
setTimeout(() => debouncedHelloCancelable.cancel(), 200); // 取消执行
setTimeout(() => debouncedHelloCancelable("Ivy"), 500); // 延迟500ms,会执行

这种方式为防抖函数提供了一个额外的控制手段,可以在需要的时候阻止函数的执行。

为什么要使用apply而不是直接调用func()?

使用apply可以确保函数在正确的上下文中执行,并传递正确的参数。apply允许我们指定this的值(即上下文)和参数数组。这在处理事件监听器等场景时非常重要,因为我们需要确保函数在触发事件的元素上下文中执行。如果直接调用func()this的值可能会指向全局对象(例如window),导致错误。

防抖和节流的区别是什么?

防抖和节流都是为了优化性能,减少函数执行频率的技术,但它们的策略不同。

  • 防抖 (Debounce): 在事件被触发后,延迟一段时间执行,如果在这段时间内事件再次被触发,则重新计时。最终只执行最后一次触发。适用于搜索建议、调整窗口大小等场景。
  • 节流 (Throttle): 在一段时间内,只允许函数执行一次。如果在该时间段内事件被多次触发,只有第一次触发会被执行。适用于滚动事件、鼠标移动等场景。

简单来说,防抖是“最后一次有效”,节流是“一段时间内只执行一次”。

防抖函数的常见应用场景有哪些?

  • 搜索建议: 用户在输入框中输入内容时,不需要每次输入都发送请求,而是等待用户停止输入一段时间后,再发送请求获取搜索建议。
  • 调整窗口大小: 在调整窗口大小的过程中,不需要每次调整都重新计算布局,而是等待调整完成后,再重新计算布局。
  • 按钮点击: 防止用户快速多次点击按钮,导致重复提交表单或执行其他操作。
  • 文本编辑器自动保存: 用户编辑文本时,不需要每次修改都保存,而是等待用户停止编辑一段时间后,再自动保存。

相关专题

更多
js正则表达式
js正则表达式

php中文网为大家提供各种js正则表达式语法大全以及各种js正则表达式使用的方法,还有更多js正则表达式的相关文章、相关下载、相关课程,供大家免费下载体验。

510

2023.06.20

js获取当前时间
js获取当前时间

JS全称JavaScript,是一种具有函数优先的轻量级,解释型或即时编译型的编程语言;它是一种属于网络的高级脚本语言,主要用于Web,常用来为网页添加各式各样的动态功能。js怎么获取当前时间呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

244

2023.07.28

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

254

2023.08.03

js是什么意思
js是什么意思

JS是JavaScript的缩写,它是一种广泛应用于网页开发的脚本语言。JavaScript是一种解释性的、基于对象和事件驱动的编程语言,通常用于为网页增加交互性和动态性。它可以在网页上实现复杂的功能和效果,如表单验证、页面元素操作、动画效果、数据交互等。

5267

2023.08.17

js删除节点的方法
js删除节点的方法

js删除节点的方法有:1、removeChild()方法,用于从父节点中移除指定的子节点,它需要两个参数,第一个参数是要删除的子节点,第二个参数是父节点;2、parentNode.removeChild()方法,可以直接通过父节点调用来删除子节点;3、remove()方法,可以直接删除节点,而无需指定父节点;4、innerHTML属性,用于删除节点的内容。

475

2023.09.01

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

206

2023.09.04

Js中concat和push的区别
Js中concat和push的区别

Js中concat和push的区别:1、concat用于将两个或多个数组合并成一个新数组,并返回这个新数组,而push用于向数组的末尾添加一个或多个元素,并返回修改后的数组的新长度;2、concat不会修改原始数组,是创建新的数组,而push会修改原数组,将新元素添加到原数组的末尾等等。本专题为大家提供concat和push相关的文章、下载、课程内容,供大家免费下载体验。

217

2023.09.14

js截取字符串的方法介绍
js截取字符串的方法介绍

JavaScript字符串截取方法,包括substring、slice、substr、charAt和split方法。这些方法可以根据具体需求,灵活地截取字符串的不同部分。在实际开发中,根据具体情况选择合适的方法进行字符串截取,能够提高代码的效率和可读性 。

218

2023.09.21

Java 项目构建与依赖管理(Maven / Gradle)
Java 项目构建与依赖管理(Maven / Gradle)

本专题系统讲解 Java 项目构建与依赖管理的完整体系,重点覆盖 Maven 与 Gradle 的核心概念、项目生命周期、依赖冲突解决、多模块项目管理、构建加速与版本发布规范。通过真实项目结构示例,帮助学习者掌握 从零搭建、维护到发布 Java 工程的标准化流程,提升在实际团队开发中的工程能力与协作效率。

10

2026.01.12

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.2万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号