前端学习15 防抖和节流

在前端开发中,处理频繁的事件触发是一项常见任务,例如窗口的 resize、页面滚动的 scroll 事件、用户输入的 keyup 或 keydown 事件等。如果不加以控制,这些事件会频繁触发,从而导致性能问题。因此,防抖(Debounce)和节流(Throttle)这两种技术就显得尤为重要。本文将详细介绍防抖和节流的概念、区别、应用场景以及实现方法。

1.防抖

1.1 防抖概念

防抖是一种延迟执行的技术。它的原理是,当事件被触发时,延迟执行事件处理函数,并且在延迟时间内如果事件再次被触发,则重新开始计时。只有当事件在指定的时间内没有再次触发,事件处理函数才会执行。这样可以避免某些高频率的操作被频繁触发,从而提高性能。

1.2 适用场景

防抖主要适用于那些在用户停止操作后才需要执行的场景,例如:

搜索输入框:用户在输入时,连续触发 keyup 事件,只有在输入结束后才发送请求。 窗口调整:用户调整浏览器窗口大小时,频繁触发 resize 事件。防抖可以确保调整结束后再执行相应操作。 表单验证:用户输入表单数据时,可以用防抖来减少频繁的验证请求。

1.3 防抖实现

function debounce(func, wait) {
    let timeout;
    return function(...args) {
        const context = this;
        clearTimeout(timeout);
        timeout = setTimeout(() => func.apply(context, args), wait);
    };
}

debounce 函数接收两个参数:func 是需要执行的函数,wait 是延迟的时间。

每次触发事件时,先清除上一次的 timeout,然后重新设置一个新的定时器。

只有在指定的 wait 时间内没有新的事件触发时,func 才会被执行。

假设我们在一个搜索输入框中使用防抖来减少请求次数:

const searchInput = document.getElementById('search');
const handleSearch = debounce(function() {
    console.log('Sending request for:', this.value);
}, 500);
 
searchInput.addEventListener('keyup', handleSearch);

在这个例子中,只有在用户停止输入 500 毫秒后,handleSearch 才会被执行,减少了不必要的请求。

2.节流(Throttle)

2.1 节流概念

节流是一种限制函数执行频率的技术。它的原理是,当事件被频繁触发时,函数会按照一定的时间间隔执行,而不是每次触发事件都执行。换句话说,在一个时间段内,只会执行一次事件处理函数。

2.2 适用场景

节流适用于需要间隔时间执行的场景,例如:

页面滚动:用户滚动页面时触发 scroll 事件,使用节流限制处理函数的执行频率。

按钮点击:防止用户短时间内多次点击同一个按钮,造成重复提交。

游戏动画:限制每秒渲染的帧数,以减少资源消耗。

2.3 节流实现

/**
 * 节流函数
 * @param {Function} func - 需要节流的函数
 * @param {number} wait - 时间间隔(毫秒),表示在这个时间间隔内最多执行一次函数
 * @returns {Function} - 返回一个节流后的函数
 */
function throttle(func, wait) {
    // 上一次执行函数的时间戳,初始值为 0
    let lastTime = 0;
 
    // 返回一个闭包函数,作为节流后的函数
    return function (...args) {
        // 获取当前时间戳
        const now = Date.now();
 
        // 如果当前时间与上一次执行时间的差值大于等于 wait,则执行函数
        if (now - lastTime >= wait) {
            // 更新上一次执行函数的时间戳
            lastTime = now;
            // 调用原始函数,并传入参数
            func.apply(this, args);
        }
    };
}

下面是一个使用节流限制 scroll 事件处理频率的例子:

// 原始的滚动事件处理函数
function handleScroll() {
    console.log('Scroll event triggered');
}
 
// 使用节流函数包装 handleScroll
const throttledScrollHandler = throttle(handleScroll, 200);
 
// 监听滚动事件,并使用节流后的函数
window.addEventListener('scroll', throttledScrollHandler);

在这个例子中,handleScroll 函数会在每次滚动时执行,但每隔一段时间只会执行一次,即使 scroll 事件在这段时间内被频繁触发。

3.综合应用

有时,我们需要在同一个项目中同时使用防抖和节流。例如,在一个输入框中进行搜索联想提示时,我们可以用节流来控制 API 请求的频率,用防抖来控制显示搜索结果的时间。

const searchInput = document.getElementById('search');
 
// 使用防抖来控制输入事件
const handleSearch = debounce(function() {
    // 使用节流来控制请求频率
    throttleSearch();
}, 300);
 
// 使用节流来控制请求频率
const throttleSearch = throttle(function() {
    console.log('Fetching search results for:', searchInput.value);
}, 1000);
 
searchInput.addEventListener('keyup', handleSearch);

在这个示例中,防抖函数用于减少输入事件的频率,只有在用户停止输入 300 毫秒后才会触发 throttleSearch。而 throttleSearch 函数会使用节流来限制请求频率,确保在 1000 毫秒内只发起一次请求。

4.总结

防抖和节流是前端优化中常用的技术,合理使用可以显著提升用户体验和系统性能。两者虽然用途不同,但都能有效地减少高频事件触发带来的性能开销。防抖适用于延迟执行的场景,节流适用于限制执行频率的场景,开发者可以根据需求选择合适的技术,甚至组合使用以达到最佳效果。

#前端学习#
全部评论

相关推荐

个人背景:- 985硕士,计算机专业,研究方向为机器学习/数据挖掘- 有推荐系统相关项目,Kaggle竞赛经历- 面试岗位:滴滴出行-算法工程师(机器学习/运筹优化方向)📝 面试全流程回顾1. 笔试(线上编程+数学)-算法题(2道,LeetCode中等偏上难度)- 动态规划:最长递增子序列变种(需优化到O(nlogn))- 图论:Dijkstra算法实现+路径还原- 数学题(概率统计+线性代数)- 贝叶斯定理应用题(拼车场景下的概率计算)- 矩阵分解(SVD)的原理与优化意义2. 技术一面(1小时)- 代码能力- 手撕:实现带权随机抽样(Reservoir Sampling变种)- 代码优化:如何减少时间复杂度?- 机器学习基础- XGBoost vs LightGBM的差异?如何选择分裂点?- 如何解决推荐系统中的冷启动问题?- 业务场景题- 滴滴拼车订单匹配如何建模?(聚类+贪心算法的取舍)3. 技术二面(1.5小时)- 项目深挖- 详细介绍Kaggle竞赛方案(特征工程、模型融合技巧)- 追问:如果数据分布偏移(如疫情前后出行规律变化),如何调整模型?- 系统设计- 设计一个实时ETA(预估到达时间)系统:- 数据源(GPS/交通路况/历史数据)- 模型选型(时序模型+在线学习)- 异常情况处理(突发拥堵如何动态调整?)- 算法发散题- 如何用算法减少司机空驶率?(转化为图的最短路径问题)4. HR面(30分钟)- 团队协作经历、抗压能力举例- 期望薪资与工作地点偏好🌟 滴滴算法团队特点业务驱动:算法直接影响亿级用户体验,成就感强技术栈前沿:时空预测、强化学习、因果推断等均有落地成长快:技术大牛密集,新人可接触核心项目🚘投递方式【内推链接】https://app.mokahr.com/m/campus_apply/didiglobal/96064?recommendCode=DSW46Dg7&hash=%23%2Fjobs#/jobs【内推码】DSW46Dg7立刻投递,快人一步,抢跑未来全流程跟进,投递的同学评论区留言,方便后续跟进,秋招加油!#实习# #应届# #春招急招# #滴滴# #滴滴出行#                                                                                   
点赞 评论 收藏
分享
评论
3
1
分享

创作者周榜

更多
牛客网
牛客企业服务