JS的防抖与节流

1|0定义

  • 节流(throttle): n 秒内只运行一次,若在 n 秒内重复触发,只有一次生效
在规定的间隔时间范围内不会重复触发回调,只有大于这个时间间隔才会触发回调,把频繁触发变为少量触发
  • 防抖(debounce): n 秒后在执行该事件,若在 n 秒内被重复触发,则重新计时

前面的所有的触发都被取消,最后一次执行在规定的时间之后才会触发,也就是说如果连续快速的触发,只会执行最后一次

2|0节流

<div> <h1>计数器<span>0</span></h1> <button>点我+1</button> </div> let span = document.querySelector('span') let btn = document.querySelector('button') let count = 0 btn.onclick = _.throttle(function(){ //节流:目前的这个函数5s执行一次 //假如这里面有很多的业务代码,可以给浏览器充分的时间去解析 count++ span.innerHTML = count console.log(1); },5000)

3|0防抖(debounce)

<p>请你输入你搜索的内容: <input type="text"></p> let inp=document.querySelector('input') inp.oninput=_.debounce(function(){ console.log('ajax'); },1000)

3|1应用场景

防抖在连续的事件,只需触发一次回调的场景有:

  • 搜索框搜索输入。只需用户最后一次输入完,再发送请求
  • 手机号、邮箱验证输入检测
  • 窗口大小resize。只需窗口调整完成后,计算窗口大小。防止重复渲染。

节流在间隔一段时间执行一次回调的场景有:

  • 滚动加载,加载更多或滚到底部监听
  • 搜索框,搜索联想功能

4|0手写简单的节流与防抖

4|1节流(throttle)

function throttle(fn, time) { var startTime = +new Date()//初始时间戳 return function () { var timeout = (+new Date() - startTime) >= time//;//判断时间间隔是否大于传入的time值,返回布尔值 if (timeout) { fn.apply(this) startTime = +new Date()//重新初始化时间戳 } } } //测试 function test() { console.log('测试'); } const throttleTest = throttle(test, 5000) window.addEventListener('scroll', throttleTest)

4|2防抖(debounce)

<input type="text" id="inp"> function getUserAction() { console.log(1); } inp.oninput = debounce(getUserAction, 1000); function debounce(fn, wait) { var timeout return function () { clearTimeout(timeout) timeout = setTimeout(fn, wait) } }

 这就实现了一个简易的防抖,指定时间内重复触发函数只会执行一次。已经很好的解决了高频重复触发的问题,但因为直接重写了oninput函数,this与event均丢失了。

先解决this指向问题,正常在oninput函数中,this指向的应该是调用函数的对象,也就是inp,因为在debounce中返回了真正重写oninput的函数,形成了一个闭包,导致this丢失。指定this指向的方法有apply与call,用法差别只在参数,call需要的是一个数组,这里就用apply。

function debounce(func, wait) { var timeout; return function () { var context = this; clearTimeout(timeout) timeout = setTimeout(function(){ func.apply(context) }, wait); } }

 解决event参数,正常在oninput函数中有一个event,指向当前的事件对象。因为debounce函数返回的是一个匿名函数,可以通过arguments属性来获取参数,该属性是一个由函数参数组成的类数组。

修改后的debounce:

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

 

 

参考:链接:https://juejin.cn/post/7030787304696315918


__EOF__

本文作者长安
本文链接https://www.cnblogs.com/jingxin01/p/16408277.html
关于博主:评论和私信会在第一时间回复。或者直接私信我。
版权声明:本博客所有文章除特别声明外,均采用 BY-NC-SA 许可协议。转载请注明出处!
声援博主:如果您觉得文章对您有帮助,可以点击文章右下角推荐一下。您的鼓励是博主的最大动力!
posted @   长安·念  阅读(167)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 震惊!C++程序真的从main开始吗?99%的程序员都答错了
· 别再用vector<bool>了!Google高级工程师:这可能是STL最大的设计失误
· 单元测试从入门到精通
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理
· 上周热点回顾(3.3-3.9)
点击右上角即可分享
微信分享提示