JS专题之节流函数

本文共 2000 字,读完只需 8 分钟

上一篇文章讲了去抖函数,然后这一篇讲同样为了优化性能,降低事件处理频率的节流函数。

一、什么是节流?

节流函数(throttle)就是让事件处理函数(handler)在大于等于执行周期时才能执行,周期之内不执行,即事件一直被触发,那么事件将会按每小段固定时间一次的频率执行。

打个比方:王者荣耀、英雄联盟、植物大战僵尸游戏中,技能的冷却时间,技能的冷却过程中,是无法使用技能的,只能等冷却时间到之后才能执行。

那什么样的场景能用到节流函数呢?
比如:

  1. 页面滚动和改变大小时需要进行业务处理,比如判断是否滑到底部,然后进行懒加载数据。
  2. 按钮被高频率地点击时,比如游戏和抢购网站。

我们通过一个简单的示意来理解:

节流函数可以用时间戳和定时器两种方式进行处理。

二、时间戳方式实现

<div class="container" id="container">正在滑动:0
</div><script>
window.onload = function() {var bodyEl = document.getElementsByTagName("body")[0]
}var count = 0;
window.onmousemove = throttle(eventHandler, 1000);function eventHandler(e) {var containerEl = document.getElementById("container");containerEl.innerHTML = "正在滑动: " + count;count++;
}function throttle(func, delay) {var delay = delay || 1000;var previousDate = new Date();var previous = previousDate.getTime();  // 初始化一个时间,也作为高频率事件判断事件间隔的变量,通过闭包进行保存。return function(args) {var context = this;var nowDate = new Date();var now = nowDate.getTime();if (now - previous >= delay) {  // 如果本次触发和上次触发的时间间隔超过设定的时间func.call(context, args);  // 就执行事件处理函数 (eventHandler)previous = now;  // 然后将本次的触发时间,作为下次触发事件的参考时间。}}
}
</script>

看时间戳实现版本的效果:

三、定时器方式实现

<div class="container" id="container">正在滑动: 0
</div><script>
window.onload = function() {var bodyEl = document.getElementsByTagName("body")[0]
}var count = 0;
window.onmousemove = throttle(eventHandler, 1000);function eventHandler(e) {var containerEl = document.getElementById("container");containerEl.innerHTML = "正在滑动: " + count;count++;
}function throttle(func, delay) {var delay = delay || 1000;var timer = null;return function(args) {var context = this;var nowDate = new Date();var now = nowDate.getTime();if (!timer) {timer = setTimeout(function() {func.call(context, args);timer = null;}, delay)}}}
</script>

看看定时器版实现版本的效果:

三、时间戳和定时器的对比分析

对比时间戳和定时器两种方式,效果上的区别主要在于:

事件戳方式会立即执行,定时器会在事件触发后延迟执行,而且事件停止触发后还会再延迟执行一次。

具体选择哪种方式取决于使用场景。underscore 把这两类场景用 leading 和 trailing 进行了表示。

四、兼容两种方式, underscore 源码实现

underscore 的源码中就同时实现了时间戳和定时器实现方式,在调用时可以自由选择要不要在间隔时间开始时(leading)执行,或是间隔时间结束后(trailing)执行。

具体看伪代码和示意图:

<div class="container" id="container">正在滑动: 0</div><div class="height"></div><script>window.onload = function() {var bodyEl = document.getElementsByTagName("body")[0]}var count = 0;// 事件处理函数function eventHandler(e) {var containerEl = document.getElementById("container");containerEl.innerHTML = "正在滑动: " + count;count++;}var _throttle = function(func, wait, options) {var context, args, result;// 定时器变量默认为 null, 是为了如果想要触发了一次后再延迟执行一次。var timeout = null;// 上一次触发事件回调的时间戳。 默认为 0 是为了方便第一次触发默认立即执行var previous = 0;// 如果没有传入 options 参数// 则将 options 参数置为空对象if (!options)options = {};var later = function() {// 如果 options.leading === false// 则每次触发回调后将 previous 置为 0, 表示下次事件触发会立即执行事件处理函数// 否则置为当前时间戳previous = options.leading === false ? 0 : +new Date();// 剩余时间跑完,执行事件,并把定时器变量置为空,如果不为空,那么剩余时间内是不会执行事件处理函数的,见 else if 那。timeout = null;result = func.apply(context, args);// 剩余时间结束,并执行完事件后,清理闭包中自由变量的内存垃圾,因为不再需要了。if (!timeout)context = args = null;};// 返回的事件回调函数return function() {// 记录当前时间戳var now = +new Date();// 第一次执行回调(此时 previous 为 0,之后 previous 值为上一次时间戳)// 并且如果程序设定第一个回调不是立即执行的(options.leading === false)// 则将 previous 值(表示上次执行的时间戳)设为 now 的时间戳(第一次触发时)// 表示刚执行过,这次就不用执行了if (!previous && options.leading === false)previous = now;// 间隔时间 和 上一次到本次事件触发回调的持续时间的时间差var remaining = wait - (now - previous);context = this;args = arguments;// 如果间隔时间还没跑完,则不会执行任何事件处理函数。// 如果超过间隔时间,就可以触发方法(remaining <= 0)// remaining > wait,表示客户端系统时间被调整过// 也会立即执行 func 函数if (remaining <= 0 || remaining > wait) {if (timeout) {clearTimeout(timeout);// 解除引用,防止内存泄露timeout = null;}// 重置前一次触发的时间戳previous = now;// result 为事件处理函数(handler)的返回值// 采用 apply 传递类数组对象 argumentsresult = func.apply(context, args);// 引用置为空,防止内存泄露if (!timeout)context = args = null;} else if (!timeout && options.trailing !== false) {// 如果 remaining > 0, 表示在间隔时间内,又触发了一次事件// 如果 trailing 为真,则会在间隔时间结束时执行一次事件处理函数(handler)// 在从触发到剩余时间跑完,会利用一个定时器执行事件处理函数,并在定时器结束时把 定时器变量置为空// 如果剩余事件内已经存在一个定时器,则不会进入本  else if 分支, 表示剩余时间已经有一个定时器在运行,该定时器会在剩余时间跑完后执行。// 如果 trailing =  false,即不需要在剩余时间跑完执行事件处理函数。// 间隔 remaining milliseconds 后触发 later 方法timeout = setTimeout(later, remaining);}// 回调返回值return result;};};window.onmousemove = _throttle(eventHandler, 1000);</script>

下面是我画的示意图:

大致总结一下代码对事件处理逻辑的影响:

  1. 如果 leading 为真,那么绿色意味着间隔时间的开始会立即执行,第一次触发也会立即执行。
  2. 如果 trailing 为真,那么从蓝紫色的竖细线后的剩余事件,会跑一个定时器,定时器在时间间隔结束时再执行一次事件处理函数。
  3. 如果 leading 不为真,那么第一次事件触发不会立即执行。
  4. 如果 trailing 不为真,最后一次事件触发后,不然再执行一次事件处理函数。

节流和去抖的常见场景

  1. 输入框打字输入完后才开始异步请求数据校验内容:去抖
  2. 下拉滚动条判断是否到底部,进行懒加载数据:去抖和节流都可以,判断是否到底的方式不同
  3. 活动网站、游戏网站,按钮被疯狂点击:节流

六、总结

去抖和节流函数都是为了降低高频率事件触发的事件处理频率,从而优化网页中大量重绘重排带来的性能问题。

其区别在于去抖会在高频率事件触发时,只执行一次,节流会在满足间隔时间后执行一次。去抖的 immediate,节流中的 leading, trailing 都是为了尽可能满足这类工具函数的不同使用场景。

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

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

相关文章

vue 2.6 插槽v-slot用法记录

v-slot用法简记用法示例匿名插槽与具名插槽插槽作用域组件使用插槽动态命名总结用法示例 vue2.6统一了插槽的语法v-slot 匿名插槽与具名插槽 在其他组件中使用child组件 <child><template v-slot:slotName>hello world</template> </child>child组…

Latex排版全解(转)

Latex排版全解 http://blog.csdn.net/langb2014/article/details/51354238转载于:https://www.cnblogs.com/yifdu25/p/8338399.html

git-ftp Can't access remote 'ft://...', exiting...问题记录

环境 服务器&#xff1a;西部数码虚拟主机 本地系统&#xff1a;windows 10 (LTSC 2019) 软件&#xff1a; Git Bash&#xff0c;gti-ftp (版本1.6.0) 问题 在使用git ftp init初始化上传代码的时候会出现 $ git ftp init fatal: Cant access remote ftp://dmkt:***dmkt.goto…

【Flutter教程】从零构建电商应用(一)

在这个系列中&#xff0c;我们将学习如何使用google的移动开发框架flutter创建一个电商应用。本文是flutter框架系列教程的第一部分&#xff0c;将学习如何安装Flutter开发环境并创建第一个Flutter应用&#xff0c;并学习Flutter应用开发中的核心概念&#xff0c;例如widget、状…

为OWA自定义快捷键

这篇短文分享一下如何为自己常用的网页添加自定义功能&#xff0c;例如添加快捷键。我这里用一个常用的网站作为范例。它是Outlook Web Access (OWA), 它的地址一般如下。我在写邮件时希望能用一些快捷键来提高工作效率&#xff0c;但系统默认自带的快捷键特别少&#xff0c;而…

数据结构 快速排序

快速排序是对冒泡排序的一种改进&#xff0c;是所有内部排序算法中平均性能最优的排序算法。其基本思想是基于分治法的&#xff1a;在待排序数组L[1...n]中任取一个元素pivot作为基准&#xff0c;从数组的两端开始扫描。设两个指示标志&#xff08;low指向起始位置&#xff0c;…

Finally语句块的运行

一、finally语句块是否一定运行&#xff1f; Java中异常捕获机制try...catch...finally块中的finally语句是不是一定会被运行&#xff1f;非常多人都说不是。当然他们的回答是正确的&#xff0c;经过试验。至少下面有两种情况下finally语句是不会被运行的&#xff1a; &#xf…

vue-cli 3.0 跨域请求代理

官方文档中指明&#xff0c;跨域请求可以通过配置vue.config.js中的devServer.proxy选项来进行配置。 这个选项配置的本质实际上就是http-proxy-middleware中间件的用法&#xff0c;和Webpack-dev-server的proxy一样。 vue-cli 3.0中介绍了两种常见的用法&#xff1a; modul…

小米人员架构调整:组建中国区,王川任总裁

12月13日上午&#xff0c;小米内部发布人员调整公开信&#xff0c;信中传达了两个重要内容&#xff1a;将销售与服务部改组为中国区&#xff0c;任命集团高级副总裁王川兼任中国区总裁。 在今年9月份&#xff0c;也就是小米上市前夕&#xff0c;雷军在一封内部信中宣布对公司组…

在 .NET 7上使用 WASM 和 WASI

WebAssembly&#xff08;WASM&#xff09;和WebAssembly System Interface&#xff08;WASI&#xff09;为开发人员开辟了新的世界。.NET 开发人员在 Blazor WebAssembly 发布时熟悉了 WASM。Blazor WebAssembly 在浏览器中基于 WebAssembly 的 .NET 运行时上运行客户端。WASI通…

Java基础 五 方法

方法 1.1 方法概述 在我们的日常生活中&#xff0c;方法可以理解为要做某件事情&#xff0c;而采取的解决办法。 如&#xff1a;小明同学在路边准备坐车来学校学习。这就面临着一件事情&#xff08;坐车到学校这件事情&#xff09;需要解决&#xff0c;解决办法呢&#xf…

django rest framework 过滤 lim分页

一.过滤 1.首先引用diango 自带的过滤配置 2.导入模块 from django_filters.rest_framework import DjangoFilterBackend from django_filters import rest_framework as filters 3.一种简单的过滤: class BookView(ModelViewSet):queryset Book.objects.all()serializer_clas…

MySQL用户及权限管理

MySQL用户及权限管理查看用户及权限查看用户及作用域&#xff08;使用范围&#xff09;查看用户权限创建用户及授权字段参数用户管理使用命令提示符登录MySQL mysql -h localhost -u root -p查看用户及权限 mysql中的用户信息和权限等都存储在一个名为mysql的数据库中。其中主…

附近有什么?8款可以查周边的App

如今科技发达的时代&#xff0c;手机的功能不仅仅只是能通讯聊天&#xff0c;而是逐渐的走进了人们的生活中。因为有了APP&#xff0c;我们的生活才更丰富&#xff0c;并且有很多是我们生活中不可缺少的软件&#xff0c;而这些软件便是根据手机中的GPS定位系统而来的。简单来说…

MyEclipse小问题与汉字处理

今天在使用MyEclipse时&#xff0c;遇到工作目录报错(如上图)&#xff0c;解决方法如下&#xff1a;找到对应工作区(查看工作区的方法为&#xff1a;单击File → Switch Workspace 即可)依次打开 .metadata文件夹 → .plugins文件夹 → org.eclipse.core.runtime文件夹 → .set…

关系数据库设计及优化原则

一直以来就想总结一下自己这么多年来在关系数据库上积累的经验。奈何自己是一个比较懒的人一直不想动手去写。扎克伯格曾说过&#xff1a;“想做一件事的话&#xff0c;最好的办法就是先开始”。索性就先写一点东西&#xff0c;这些东西不会太长&#xff0c;自然也不会包括太多…

java B2B2C springmvc mybatis电子商务平台源码-消息队列之RocketMQ

RocketMQ出自阿里公司的开源产品&#xff0c;用 Java 语言实现&#xff0c;在设计时参考了 Kafka&#xff0c;并做出了自己的一些改进&#xff0c;消息可靠性上比 Kafka 更好。RocketMQ在阿里集团被广泛应用在订单&#xff0c;交易&#xff0c;充值&#xff0c;流计算&#xff…

VSCode同步设置

2022/4/1 更新 刚刚发现还有人在看这篇文章&#xff0c;这里更新一下&#xff0c;VSCode 从1.48版本开始已经内置了同步功能&#xff0c;可以不用再使用Settings Sync插件了。 点击左下角的用户或者设置的 Sign in to Sync Setting&#xff0c;使用GitHub或者Microsoft账户登…

配置三台服务器组成的ELK集群(二)

上一篇里主要是介绍了ES和ES-Head的安装过程&#xff0c;这一篇继续介绍ELK集群的其他核心组件安装过程。 五、安装Logstash&#xff1a; 本案的Logstash安装在10.113.130.117上&#xff1b;燃鹅&#xff0c;Logstash也可以利用多台组成集群&#xff0c;如果未来单台处理不过来…

Discuz X3.2源码解析 discuz_application类(转自百度)

discuz_application在/source/class/discuz/discuz_application.php中。 discuz_application继承自抽象类discuz_base discuz_application主要实现对运行环境、配置、输入、输出、数据库、设置、用户、session、移动模块、计划任务、手机预览等方面的初始化。 instance()函数来…