JavaScript 中的防抖和节流,它们的区别是什么,以及如何实现?

在前端开发中,防抖(Debounce)节流(Throttle)是两种常用的优化高频率事件处理的技术。

它们能够有效减少事件处理函数的执行次数,从而提升页面性能和用户体验。

下面将详细解释这两种技术的概念、区别、实现方法以及在日常开发中的使用建议和注意事项。

一、防抖(Debounce)

概念:防抖是指在事件被触发后,等待一定的时间,如果在这段时间内没有再次触发该事件,则执行一次事件处理函数;如果在等待期间内事件再次被触发,则重新计时。

应用场景

  • 窗口调整大小(resize):用户调整窗口大小时,频繁触发resize事件,使用防抖可以只在用户停止调整后执行一次处理逻辑。
  • 输入框实时搜索:用户在输入框中输入内容时,频繁触发输入事件,使用防抖可以只在用户停止输入一段时间后才发送搜索请求。

实现

/*** 防抖函数* @param {Function} func - 需要防抖处理的函数* @param {number} wait - 等待时间(毫秒)* @param {boolean} immediate - 是否立即执行* @returns {Function} - 返回防抖处理后的函数*/
function debounce(func, wait, immediate = false) {let timeout; // 定义一个定时器变量return function(...args) {const context = this; // 保存当前上下文const later = () => {timeout = null; // 清空定时器if (!immediate) func.apply(context, args); // 如果不是立即执行,则执行函数};const callNow = immediate && !timeout; // 判断是否立即执行clearTimeout(timeout); // 清除之前的定时器timeout = setTimeout(later, wait); // 设置新的定时器if (callNow) func.apply(context, args); // 如果需要立即执行,则执行函数};
}// 使用示例
const handleResize = debounce(() => {console.log('窗口大小调整完成');
}, 300);window.addEventListener('resize', handleResize);

代码说明

  • debounce 函数接收一个函数 func、等待时间 wait 和一个布尔值 immediate
  • 返回一个新的函数,在该函数被调用时,会设置一个定时器 timeout,在 wait 时间后执行 func
  • 如果在 wait 时间内再次调用该函数,会清除之前的定时器并重新设置,从而实现防抖效果。
  • 参数 immediate 控制是否在第一次触发时立即执行函数。
二、节流(Throttle)

概念:节流是指在一定的时间间隔内,只执行一次事件处理函数。无论事件触发频率多高,处理函数都会按照固定的时间间隔执行。

应用场景

  • 滚动事件(scroll):用户滚动页面时,频繁触发scroll事件,使用节流可以限制处理函数的执行频率,提升性能。
  • 鼠标移动事件(mousemove):在拖拽操作中,频繁触发mousemove事件,使用节流可以减少处理次数。

实现

/*** 节流函数* @param {Function} func - 需要节流处理的函数* @param {number} wait - 时间间隔(毫秒)* @param {Object} options - 可选参数* @param {boolean} options.leading - 是否在开始时执行* @param {boolean} options.trailing - 是否在结束时执行* @returns {Function} - 返回节流处理后的函数*/
function throttle(func, wait, options = {}) {let timeout = null; // 定时器变量let lastArgs = null; // 上一次的参数let lastThis = null; // 上一次的上下文let lastCallTime = 0; // 上一次调用的时间const later = () => {lastCallTime = options.leading === false ? 0 : Date.now();timeout = null;func.apply(lastThis, lastArgs);if (!timeout) lastArgs = lastThis = null;};return function(...args) {const now = Date.now();if (!lastCallTime && options.leading === false) lastCallTime = now;const remaining = wait - (now - lastCallTime);lastArgs = args;lastThis = this;if (remaining <= 0 || remaining > wait) {if (timeout) {clearTimeout(timeout);timeout = null;}lastCallTime = now;func.apply(lastThis, lastArgs);} else if (!timeout && options.trailing !== false) {timeout = setTimeout(later, remaining);}};
}// 使用示例
const handleScroll = throttle(() => {console.log('页面滚动');
}, 200);window.addEventListener('scroll', handleScroll);

代码说明

  • throttle 函数接收一个函数 func、时间间隔 wait 和一个可选参数对象 options
  • 返回一个新的函数,在该函数被调用时,会根据时间间隔 wait 来决定是否执行 func
  • 参数 options.leading 控制是否在开始时立即执行,options.trailing 控制是否在结束时执行。
  • 通过记录上一次调用的时间和设置定时器,实现节流效果。
三、防抖与节流的区别
特性防抖(Debounce)节流(Throttle)
执行时机在事件停止触发后的一段时间内执行一次在固定的时间间隔内执行一次
适用场景输入框实时搜索、窗口调整大小等需要等待用户停止操作的场景滚动事件、鼠标移动事件等需要限制执行频率的场景
实现方式通过不断重置定时器,确保只有在最后一次触发后的一段时间内没有新的触发才执行通过记录上次执行时间,控制函数的执行频率
四、日常开发中的使用建议
  1. 合理选择防抖和节流

    • 对于需要等待用户停止操作的事件(如输入、窗口调整),使用防抖更为合适。
    • 对于需要限制执行频率的事件(如滚动、拖拽),使用节流更为合适。
  2. 优化用户体验

    • 在高频率触发的事件中,合理使用防抖和节流,可以避免因频繁执行导致的页面卡顿或性能问题。
    • 例如,在搜索框中输入内容时,使用防抖可以减少不必要的搜索请求,提高响应速度。
  3. 结合实际需求调整参数

    • 根据具体业务场景,调整防抖和节流的等待时间或时间间隔,以达到最佳效果。
    • 例如,对于快速滚动的页面,可以将节流的时间间隔设置为100ms,以平衡性能和响应速度。
五、实际开发过程中需要注意的点
  1. 内存管理

    • 确保在组件销毁或不再需要监听事件时,移除相应的事件监听器,防止内存泄漏。
    • 例如,使用 removeEventListener 移除绑定的防抖或节流函数。
  2. 函数上下文和参数传递

    • 在实现防抖和节流时,注意保持原函数的上下文 (this) 和参数,确保函数执行时的正确性。
    • 例如,使用 func.apply(this, args) 来调用原函数。
  3. 兼容性和性能

    • 在不同浏览器和设备上测试防抖和节流的效果,确保兼容性。
    • 避免在节流或防抖函数中执行过于复杂的逻辑,以免影响性能。
  4. 组合使用

    • 在某些复杂场景下,可能需要同时使用防抖和节流,根据具体需求灵活组合使用。

示例:在一个需要同时限制滚动事件频率和搜索输入延迟的场景中,可以分别对滚动事件使用节流,对搜索输入使用防抖。

// 节流处理滚动事件
const handleScrollThrottled = throttle(() => {console.log('滚动事件节流处理');
}, 200);window.addEventListener('scroll', handleScrollThrottled);// 防抖处理搜索输入
const handleSearchDebounced = debounce((query) => {console.log(`搜索内容: ${query}`);
}, 300);searchInput.addEventListener('input', (e) => {handleSearchDebounced(e.target.value);
});

通过合理地组合使用防抖和节流,可以在不同的事件场景中达到最佳的性能优化效果。

防抖和节流是前端开发中优化高频率事件处理的重要技术。

理解它们的概念、区别及实现方法,并在实际开发中合理应用,可以显著提升页面性能和用户体验。

在实际应用中,需根据具体场景选择合适的优化策略,并注意内存管理、函数上下文和参数传递等问题,以确保代码的健壮性和可维护性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/70698.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

推荐一个免费的、开源的大数据工程学习教程

在当今信息爆炸的时代&#xff0c;每一个企业都会产生大量的数据&#xff0c;而大数据也已经成为很多企业发展的重要驱动力&#xff0c;然而如何有效得处理和分析这些海量的数据&#xff0c;却是一个非常有挑战的技术。 今天推荐一个免费的数据工程教程&#xff0c;带你系统化…

2月10日QT

作业> 将文本编辑器功能完善 include "widget.h" #include "ui_widget.h" #include <QMessageBox> //消息对话框类 #include <QFontDialog> //字体类对话框 #include <QFont> //字体类 #include <QColorDialog> //颜…

ECG分析0210

指标计算方法 1. HR (心率&#xff0c;Heart Rate)&#xff1a; 心率是每分钟心跳的次数。它通常通过计算RR间期&#xff08;即两次R波之间的时间间隔&#xff09;来获得。 计算方法&#xff1a; 首先&#xff0c;检测到R波的位置&#xff08;例如通过find_peaks函数检测&a…

【Java】多线程和高并发编程(四):阻塞队列(上)基础概念、ArrayBlockingQueue

文章目录 四、阻塞队列1、基础概念1.1 生产者消费者概念1.2 JUC阻塞队列的存取方法 2、ArrayBlockingQueue2.1 ArrayBlockingQueue的基本使用2.2 生产者方法实现原理2.2.1 ArrayBlockingQueue的常见属性2.2.2 add方法实现2.2.3 offer方法实现2.2.4 offer(time,unit)方法2.2.5 p…

【Java】多线程和高并发编程(三):锁(下)深入ReentrantReadWriteLock

文章目录 4、深入ReentrantReadWriteLock4.1 为什么要出现读写锁4.2 读写锁的实现原理4.3 写锁分析4.3.1 写锁加锁流程概述4.3.2 写锁加锁源码分析4.3.3 写锁释放锁流程概述&释放锁源码 4.4 读锁分析4.4.1 读锁加锁流程概述4.4.1.1 基础读锁流程4.4.1.2 读锁重入流程4.4.1.…

【R语言】相关系数

一、cor()函数 cor()函数是R语言中用于计算相关系数的函数&#xff0c;相关系数用于衡量两个变量之间的线性关系强度和方向。 常见的相关系数有皮尔逊相关系数&#xff08;Pearson correlation coefficient&#xff09;、斯皮尔曼秩相关系数&#xff08;Spearmans rank corre…

Web - CSS3过渡与动画

过渡 基本使用 transition过渡属性是css3浓墨重彩的特性&#xff0c;过渡可以为一个元素在不同样式之间变化自动添加补间动画。 过渡从kIE10开始兼容&#xff0c;移动端兼容良好&#xff0c;网页上的动画特效基本都是由JavaScript定时器实现的&#xff0c;现在逐步改为css3过…

Unity 高度可扩展的技能与多 Buff 框架详解

一、框架设计 1.1 核心思想 组件化设计: 将技能和 Buff 抽象为可复用的组件&#xff0c;通过组合不同的组件实现复杂的效果。 数据驱动: 使用 ScriptableObject 或 JSON 等数据格式定义技能和 Buff 的属性&#xff0c;方便配置和修改。 事件驱动: 利用 Unity 的事件系统或自…

编译和链接【一】

文章目录 编译和链接【一】从翻译单元到二进制文件 编译和链接【一】 在我大一的时候&#xff0c; 我使用VC6.0对C语言程序进行编译链接和运行 &#xff0c; 然后我接触了VS&#xff0c; VS code等众多IDE&#xff0c; 这些IDE界面友好&#xff0c; 使用方便&#xff0c; 例如…

图像锐化(QT)

如果不使用OpenCV&#xff0c;我们可以直接使用Qt的QImage类对图像进行像素级操作来实现锐化。锐化算法的核心是通过卷积核&#xff08;如拉普拉斯核&#xff09;对图像进行处理&#xff0c;增强图像的边缘和细节。 以下是一个完整的Qt应用程序示例&#xff0c;展示如何使用Qt…

迅雷下载的原理和使用协议的分析

迅雷作为一款广泛使用的下载工具&#xff0c;其核心原理是通过整合多种下载协议和资源分发技术来提升下载速度。以下是对其原理及协议的详细分析&#xff1a; 一、迅雷下载的核心原理 多协议混合下载&#xff08;P2SP&#xff09; P2SP&#xff08;Peer-to-Server-Peer&#xf…

【动手学运动规划】5.4 二次规划问题:QP优化

站在天堂看地狱&#xff0c;人生就像情景剧&#xff1b;站在地狱看天堂&#xff0c;为谁辛苦为谁忙。 —武林外传 白展堂 &#x1f3f0;代码及环境配置&#xff1a;请参考 环境配置和代码运行! 在运动规划算法中, QP优化是非常常见的优化问题形式, 本节我们将进行介绍. 5.4.1…

Linux: ASoC 声卡硬件参数的设置过程简析

文章目录 1. 前言2. ASoC 声卡设备硬件参数2.1 将 DAI、Machine 平台的硬件参数添加到声卡2.2 打开 PCM 流时将声卡硬件参数配置到 PCM 流2.3 应用程序对 PCM 流参数进行修改调整 1. 前言 限于作者能力水平&#xff0c;本文可能存在谬误&#xff0c;因此而给读者带来的损失&am…

ansible使用学习

一、查询手册 1、官网 ansible官网地址&#xff1a;https://docs.ansible.com 模块查看路径&#xff1a;https://docs.ansible.com/ansible/latest/collections/ansible/builtin/index.html#plugins-in-ansible-builtin 2、命令 ansible-doc -s command二、相关脚本 1、服务…

jmap使用

常用命令 jmap -heap PID jmap -histo PID | head -20 jmap -dump:formatb,fileheap_dump.hprof PID jmap 是 Java 开发工具包&#xff08;JDK&#xff09;提供的一个命令行工具&#xff0c;用于生成 Java 进程的内存映射信息。它可以帮助开发者分析 Java 堆内存的使用情况…

RabbitMQ 如何设置限流?

RabbitMQ 的限流&#xff08;流量控制&#xff09;主要依赖于 QoS&#xff08;Quality of Service&#xff09; 机制&#xff0c;即 prefetch count 参数。这个参数控制每个消费者一次最多能获取多少条未确认的消息&#xff0c;从而避免某个消费者被大量消息压垮。 1. RabbitMQ…

第四十八章:黄山之行:与小一的奇妙冒险

自从小泽泽满月酒过后&#xff0c;小冷一家的生活又恢复了往日的温馨与忙碌。小泽泽在家人的悉心照料下茁壮成长&#xff0c;而小冷和小颖也在工作与家庭之间努力平衡着。2024 年 11 月&#xff0c;秋意正浓&#xff0c;山林间五彩斑斓&#xff0c;空气中弥漫着清爽的气息。小冷…

基于 SpringBoot 和 Vue 的智能腰带健康监测数据可视化平台开发(文末联系,整套资料提供)

基于 SpringBoot 和 Vue 的智能腰带健康监测数据可视化平台开发 一、系统介绍 随着人们生活水平的提高和健康意识的增强&#xff0c;智能健康监测设备越来越受到关注。智能腰带作为一种新型的健康监测设备&#xff0c;能够实时采集用户的腰部健康数据&#xff0c;如姿势、运动…

2025.2.8 寒假综合训练赛2题解

A. 博弈 Link&#xff1a;P1290 欧几里德的游戏 博弈类的题目&#xff0c;首先考虑找找有什么性质&#xff0c;从而找到“必胜态”和“必败态”。 其中&#xff0c;面对“必胜态”不一定取胜&#xff08;看个人操作的好坏&#xff09;&#xff0c;但面对“必败态”一定输&am…

docker离线安装及部署各类中间件(x86系统架构)

前言&#xff1a;此文主要针对需要在x86内网服务器搭建系统的情况 一、docker离线安装 1、下载docker镜像 https://download.docker.com/linux/static/stable/x86_64/ 版本&#xff1a;docker-23.0.6.tgz 2、将docker-23.0.6.tgz 文件上传到服务器上面&#xff0c;这里放在…