news 2026/1/22 6:42:43

函数节流与防抖:减少不必要的函数调用

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
函数节流与防抖:减少不必要的函数调用

在前端开发中,我们经常会遇到一些频繁触发的事件,例如窗口滚动、鼠标移动、输入框输入等。如果在这些事件的处理函数中执行一些复杂的操作,就会导致页面性能下降,甚至出现卡顿现象。函数节流(Throttle)和防抖(Debounce)就是两种可以有效减少不必要函数调用的技术,它们能够优化代码性能,提升用户体验。

什么是函数节流和防抖

函数节流(Throttle)

函数节流是指在一定时间内,只执行一次函数。就好比水龙头放水,我们可以控制它每隔一段时间放一次水,而不是一直开着让水不断流出。在实际应用中,函数节流常用于限制一些频繁触发的事件,如滚动加载、按钮点击等。

函数防抖(Debounce)

函数防抖是指在一定时间内,如果再次触发相同的函数调用,则重新计时,直到在这个时间内没有再次触发,才执行函数。可以把它想象成电梯关门,如果有人不断进出电梯,电梯门就会一直延迟关闭,直到一段时间内没有人再进出,电梯门才会关闭。函数防抖常用于搜索框输入提示、窗口大小改变等场景。

函数节流的实现原理与代码示例

实现原理

函数节流的实现原理是通过记录上一次函数执行的时间,当再次触发函数时,判断当前时间与上一次执行时间的差值是否大于设定的时间间隔,如果大于则执行函数,并更新上一次执行时间。

代码示例
functionthrottle(func,delay){letlastTime=0;returnfunction(){constnow=Date.now();if(now-lastTime>delay){func.apply(this,arguments);lastTime=now;}}}// 使用示例functionhandleScroll(){console.log('Scroll event triggered');}constthrottledScroll=throttle(handleScroll,500);window.addEventListener('scroll',throttledScroll);
图表展示

下面是一个简单的图表,展示了函数节流的执行过程:

时间(ms)事件触发函数是否执行
0触发
200触发
400触发
600触发
800触发

函数防抖的实现原理与代码示例

实现原理

函数防抖的实现原理是通过设置一个定时器,当触发函数时,先清除之前的定时器,然后重新设置一个新的定时器。如果在定时器时间内再次触发函数,则重复上述操作,直到定时器时间结束,才执行函数。

代码示例
functiondebounce(func,delay){lettimer=null;returnfunction(){if(timer){clearTimeout(timer);}timer=setTimeout(()=>{func.apply(this,arguments);},delay);}}// 使用示例functionhandleInput(){console.log('Input event triggered');}constdebouncedInput=debounce(handleInput,300);constinputElement=document.getElementById('input');inputElement.addEventListener('input',debouncedInput);
图表展示

下面是一个简单的图表,展示了函数防抖的执行过程:

时间(ms)事件触发定时器状态函数是否执行
0触发设置 300ms 定时器
100触发清除原定时器,设置新的 300ms 定时器
200触发清除原定时器,设置新的 300ms 定时器
500无触发定时器结束

节流和防抖的应用场景

节流的应用场景
  • 滚动加载:当用户滚动页面时,会不断触发滚动事件。使用函数节流可以限制滚动事件的处理函数在一定时间内只执行一次,避免频繁请求数据,提高性能。
functionloadMoreData(){// 模拟加载更多数据console.log('Loading more data...');}constthrottledLoadMore=throttle(loadMoreData,1000);window.addEventListener('scroll',throttledLoadMore);
  • 按钮点击:对于一些按钮点击事件,如点赞、提交表单等,如果用户频繁点击,可能会导致重复请求。使用函数节流可以限制按钮点击事件在一定时间内只执行一次。
functionsubmitForm(){// 模拟提交表单console.log('Form submitted');}constthrottledSubmit=throttle(submitForm,2000);constsubmitButton=document.getElementById('submit');submitButton.addEventListener('click',throttledSubmit);
防抖的应用场景
  • 搜索框输入提示:当用户在搜索框输入内容时,会不断触发输入事件。使用函数防抖可以在用户输入完成后一段时间内没有再次输入时,才发送请求获取搜索提示,减少不必要的请求。
functiongetSearchSuggestions(){// 模拟获取搜索提示console.log('Getting search suggestions...');}constdebouncedSearch=debounce(getSearchSuggestions,300);constsearchInput=document.getElementById('search');searchInput.addEventListener('input',debouncedSearch);
  • 窗口大小改变:当用户调整浏览器窗口大小时,会不断触发窗口大小改变事件。使用函数防抖可以在用户停止调整窗口大小一段时间后,才执行相应的处理函数,避免频繁重新布局和渲染。
functionhandleWindowResize(){// 模拟处理窗口大小改变事件console.log('Window resized');}constdebouncedResize=debounce(handleWindowResize,500);window.addEventListener('resize',debouncedResize);

节流和防抖的优化方案

节流的优化方案
  • 立即执行版节流:在某些场景下,我们希望函数在第一次触发时立即执行,而不是等待时间间隔。可以通过添加一个参数来实现立即执行版节流。
functionthrottle(func,delay,immediate=false){letlastTime=0;lettimer=null;returnfunction(){constnow=Date.now();if(immediate&&!lastTime){func.apply(this,arguments);lastTime=now;}elseif(now-lastTime>delay){if(timer){clearTimeout(timer);timer=null;}func.apply(this,arguments);lastTime=now;}elseif(!timer){timer=setTimeout(()=>{func.apply(this,arguments);lastTime=Date.now();timer=null;},delay-(now-lastTime));}}}
防抖的优化方案
  • 立即执行版防抖:在某些场景下,我们希望函数在第一次触发时立即执行,然后在一段时间内不再执行。可以通过添加一个参数来实现立即执行版防抖。
functiondebounce(func,delay,immediate=false){lettimer=null;returnfunction(){constcontext=this;constargs=arguments;if(timer){clearTimeout(timer);}if(immediate){constcallNow=!timer;timer=setTimeout(()=>{timer=null;},delay);if(callNow){func.apply(context,args);}}else{timer=setTimeout(()=>{func.apply(context,args);},delay);}}}

总结

函数节流和防抖是两种非常实用的前端性能优化技术,它们可以有效减少不必要的函数调用,提高页面性能和用户体验。在实际开发中,我们需要根据具体的应用场景选择合适的技术。如果需要限制函数在一定时间内只执行一次,就使用函数节流;如果需要在用户停止操作一段时间后才执行函数,就使用函数防抖。同时,我们还可以根据需求对节流和防抖函数进行优化,实现立即执行等功能。通过合理运用函数节流和防抖,我们可以让我们的代码更加高效、稳定。

避坑要点

  • 时间间隔的选择:在使用函数节流和防抖时,时间间隔的选择非常重要。如果时间间隔设置得太小,可能达不到减少函数调用的效果;如果时间间隔设置得太大,可能会影响用户体验。需要根据具体的应用场景进行合理调整。
  • this 指向问题:在使用节流和防抖函数时,要注意 this 指向问题。由于使用了闭包,this 可能会指向错误的对象。可以使用 apply 或 call 方法来确保 this 指向正确。
  • 定时器管理:在实现防抖函数时,要确保定时器的正确管理。每次触发函数时,都要清除之前的定时器,避免出现多个定时器同时存在的情况。
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/1/19 15:57:46

ECU底层开发中UDS诊断报文解析实战案例

ECU底层开发实战:从CAN报文到UDS诊断响应的完整链路拆解你有没有遇到过这样的场景?诊断仪连上ECU,发送一个读取VIN码的请求——“22 F1 90”,结果返回“7F 22 11”:服务不支持。你一头雾水:代码里明明写了H…

作者头像 李华
网站建设 2026/1/21 4:03:10

死锁:线程卡死不是偶然,而是设计问题

网罗开发(小红书、快手、视频号同名)大家好,我是 展菲,目前在上市企业从事人工智能项目研发管理工作,平时热衷于分享各种编程领域的软硬技能知识以及前沿技术,包括iOS、前端、Harmony OS、Java、Python等方…

作者头像 李华
网站建设 2026/1/18 7:28:49

PyTorch-CUDA-v2.6镜像部署TTS语音合成模型全过程

PyTorch-CUDA-v2.6镜像部署TTS语音合成模型全过程 在智能语音助手、有声读物自动生成和无障碍交互系统日益普及的今天,如何快速、稳定地将训练好的TTS(Text-to-Speech)模型投入实际运行,已成为许多AI团队面临的关键挑战。一个常见…

作者头像 李华
网站建设 2026/1/20 14:25:07

RS485和RS232通信稳定性实测数据完整示例

RS485 vs RS232:谁才是工业通信的“扛把子”?实测数据告诉你真相在一次现场调试中,我的同事老李遇到了一个经典问题:他负责的楼宇监控系统每隔几小时就会丢一包数据。排查了软件、协议、电源,最后发现根源竟然是——用…

作者头像 李华
网站建设 2026/1/21 20:38:43

PyTorch-CUDA-v2.6镜像支持哪些Python版本?官方说明解读

PyTorch-CUDA-v2.6 镜像支持哪些 Python 版本?官方构建逻辑深度解析 在现代 AI 开发中,一个稳定的训练环境往往比模型结构本身更早成为瓶颈。你有没有经历过这样的场景:刚从同事那里拿到一份“完美可运行”的代码,在自己机器上却…

作者头像 李华
网站建设 2026/1/18 22:44:33

法语教程资源合集

M 梅尔特伊1080法语中字2025 文件大小: 2.8GB内容特色: 2025法语剧情大片1080p中字收藏版适用人群: 法语学习者/文艺片爱好者/收藏党核心价值: 原汁原味法语对白精准中文字幕一次看爽下载链接: https://pan.quark.cn/s/a51378a9c657 【语言学习】001 有道法语 文件大小: 12.…

作者头像 李华