详解JavaScript节流函数中的Throttle


Posted in Javascript onJuly 16, 2016

首先我们来了解下什么是Throttle 

    1. 定义

如果将水龙头拧紧直到水是以水滴的形式流出,那你会发现每隔一段时间,就会有一滴水流出。

也就是会说预先设定一个执行周期,当调用动作的时刻大于等于执行周期则执行该动作,然后进入下一个新周期。

      接口定义:

* 频率控制 返回函数连续调用时,action 执行频率限定为 次 / delay
* @param delay {number} 延迟时间,单位毫秒
* @param action {function} 请求关联函数,实际应用需要调用的函数
* @return {function} 返回客户调用函数
*/
throttle(delay,action)

    2. 简单实现

var throttle = function(delay, action){
 var last = 0return function(){
 var curr = +new Date()
 if (curr - last > delay){
  action.apply(this, arguments)
  last = curr 
 }
 }
}

下面我仔细解释一下这个节流函数。

在浏览器 DOM 事件里面,有一些事件会随着用户的操作不间断触发。比如:重新调整浏览器窗口大小(resize),浏览器页面滚动(scroll),鼠标移动(mousemove)。也就是说用户在触发这些浏览器操作的时候,如果脚本里面绑定了对应的事件处理方法,这个方法就不停的触发。

这并不是我们想要的,因为有的时候如果事件处理方法比较庞大,DOM 操作比如复杂,还不断的触发此类事件就会造成性能上的损失,导致用户体验下降(UI 反映慢、浏览器卡死等)。所以通常来讲我们会给相应事件添加延迟执行的逻辑。

通常来说我们用下面的代码来实现这个功能:

var COUNT = 0;
function testFn() { console.log(COUNT++); }
// 浏览器resize的时候
// 1. 清除之前的计时器
// 2. 添加一个计时器让真正的函数testFn延后100毫秒触发
window.onresize = function () {
 var timer = null;
 clearTimeout(timer);
 
 timer = setTimeout(function() {
  testFn();
 }, 100);
};

细心的同学会发现上面的代码其实是错误的,这是新手会犯的一个问题:setTimeout 函数返回值应该保存在一个相对全局变量里面,否则每次 resize 的时候都会产生一个新的计时器,这样就达不到我们发的效果了

于是我们修改了代码:

var timer = null;
window.onresize = function () {
 clearTimeout(timer);
 timer = setTimeout(function() {
  testFn();
 }, 100);
};

这时候代码就正常了,但是又多了一个新问题 —— 产生了一个全局变量 timer。这是我们不想见到的,如果这个页面还有别的功能也叫 timer 不同的代码之前就是产生冲突。为了解决这个问题我们要用 JavaScript 的一个语言特性:闭包 closures 。相关知识读者可以去 MDN 中了解,改造后的代码如下:

/**
 * 函数节流方法
 * @param Function fn 延时调用函数
 * @param Number delay 延迟多长时间
 * @return Function 延迟执行的方法
 */
var throttle = function (fn, delay) {
 var timer = null;
 
 return function () {
  clearTimeout(timer);
  timer = setTimeout(function() {
   fn();
  }, delay);
 }
};
window.onresize = throttle(testFn, 200, 1000);

我们用一个闭包函数(throttle节流)把 timer 放在内部并且返回延时处理函数,这样以来 timer 变量对外是不可见的,但是内部延时函数触发时还可以访问到 timer 变量。

当然这种写法对于新手来说不好理解,我们可以变换一种写法来理解一下:

var throttle = function (fn, delay) {
 var timer = null;
 
 return function () {
  clearTimeout(timer);
  timer = setTimeout(function() {
   fn();
  }, delay);
 }
};
 
var f = throttle(testFn, 200);
window.onresize = function () {
 f();
};

这里主要了解一点:throttle 被调用后返回的 function 才是真正的 onresize 触发时需要调用的函数

现在看起来这个方法已经接近完美了,然而实际使用中并非如此。举个例子:

如果用户 不断的 resize 浏览器窗口大小,这时延迟处理函数一次都不会执行

于是我们又要添加一个功能:当用户触发 resize 的时候应该 在某段时间 内至少触发一次,既然是在某段时间内,那么这个判断条件就可以取当前的时间毫秒数,每次函数调用把当前的时间和上一次调用时间相减,然后判断差值如果大于 某段时间 就直接触发,否则还是走 timeout 的延迟逻辑。

下面的代码里面需要指出的是:

previous 变量的作用和 timer 类似,都是记录上一次的标识,必须是相对的全局变量
如果逻辑流程走的是“至少触发一次”的逻辑,那么函数调用完成需要把 previous 重置成当前时间,简单来说就是:相对于下一次的上一次其实就是当前

/**
 * 函数节流方法
 * @param Function fn 延时调用函数
 * @param Number delay 延迟多长时间
 * @param Number atleast 至少多长时间触发一次
 * @return Function 延迟执行的方法
 */
var throttle = function (fn, delay, atleast) {
 var timer = null;
 var previous = null;
 
 return function () {
  var now = +new Date();
 
  if ( !previous ) previous = now;
 
  if ( now - previous > atleast ) {
   fn();
   // 重置上一次开始时间为本次结束时间
   previous = now;
  } else {
   clearTimeout(timer);
   timer = setTimeout(function() {
    fn();
   }, delay);
  }
 }
};

实践:

我们模拟一个窗口 scroll 时节流的场景,也就是说当用户滚动页面向下的时候我们需要节流执行一些方法,比如:计算 DOM 位置等需要连续操作 DOM 元素的动作

完整代码如下:

<!DOCTYPE html>
<html lang="en">
<head>
 <meta charset="UTF-8">
 <title>throttle</title>
</head>
<body>
 <div style="height:5000px">
  <div id="demo" style="position:fixed;"></div>
 </div>
 <script>
 var COUNT = 0, demo = document.getElementById('demo');
 function testFn() {demo.innerHTML += 'testFN 被调用了 ' + ++COUNT + '次<br>';}
 
 var throttle = function (fn, delay, atleast) {
  var timer = null;
  var previous = null;
 
  return function () {
   var now = +new Date();
 
   if ( !previous ) previous = now;
   if ( atleast && now - previous > atleast ) {
    fn();
    // 重置上一次开始时间为本次结束时间
    previous = now;
    clearTimeout(timer);
   } else {
    clearTimeout(timer);
    timer = setTimeout(function() {
     fn();
     previous = null;
    }, delay);
   }
  }
 };
 window.onscroll = throttle(testFn, 200);
 // window.onscroll = throttle(testFn, 500, 1000);
 </script>
</body>
</html>

我们用两个 case 来测试效果,分别是添加至少触发 atleast 参数和不添加:

// case 1
window.onscroll = throttle(testFn, 200);
// case 2
window.onscroll = throttle(testFn, 200, 500);

case 1 的表现为:在页面滚动的过程(不能停止)中 testFN 不会被调用,直到停止的时候会调用一次,也就是说执行的是 throttle 里面 最后 一个 setTimeout ,效果如图:

 详解JavaScript节流函数中的Throttle

case 2 的表现为:在页面滚动的过程(不能停止)中 testFN 第一次会延迟 500ms 执行(来自至少延迟逻辑),后来至少每隔 500ms 执行一次,效果如图

 详解JavaScript节流函数中的Throttle

如上展示,我们要实现的效果已经介绍完毕并奉上了示例,望对有需要的朋友有所帮助。后续的一些辅助性优化读者可以自己琢磨,如:函数 this 指向,返回值保存等。总之仔仔细细理解一下这个过程感觉真好!

Javascript 相关文章推荐
jQuery焦点控制图层展示延迟隐藏的方法
Mar 09 Javascript
jQuery使用模式窗口实现在主页面和子页面中互相传值的方法
Mar 01 Javascript
JavaScript基础教程——入门必看篇
May 20 Javascript
深入理解Node.js的HTTP模块
Oct 12 Javascript
Vue2.x中的父组件传递数据至子组件的方法
May 01 Javascript
ztree实现权限横向显示功能
May 20 Javascript
Bootstrap提示框效果的实例代码
Jul 12 Javascript
layer实现关闭弹出层刷新父界面功能详解
Nov 15 Javascript
详解微信图片防盗链“此图片来自微信公众平台 未经允许不得引用”的解决方案
Apr 04 Javascript
使用vue脚手架(vue-cli)搭建一个项目详解
May 09 Javascript
如何通过JS实现转码与解码
Feb 21 Javascript
es6函数中的作用域实例分析
Apr 18 Javascript
很棒的js选项卡切换效果
Jul 15 #Javascript
轻松5句话解决JavaScript的作用域
Jul 15 #Javascript
jQuery EasyUI基础教程之EasyUI常用组件(推荐)
Jul 15 #Javascript
IE下JS保存图片的简单实例
Jul 15 #Javascript
jQuery 3.0中存在问题及解决办法
Jul 15 #Javascript
JavaScript6 let 新语法优势介绍
Jul 15 #Javascript
简单实现轮播图效果的实例
Jul 15 #Javascript
You might like
PHP开发中的错误收集,不定期更新。
2011/02/03 PHP
pdo中使用参数化查询sql
2011/08/11 PHP
php时区转换转换函数
2014/01/07 PHP
destoon实现调用自增数字从1开始的方法
2014/08/21 PHP
php实现可用于mysql,mssql,pg数据库操作类
2014/12/13 PHP
Symfony2创建基于域名的路由相关示例
2016/11/14 PHP
php判断目录存在的简单方法
2019/09/26 PHP
js 数组克隆方法 小结
2010/03/20 Javascript
Node.js中child_process实现多进程
2015/02/03 Javascript
jQuery+Ajax请求本地数据加载商品列表页并跳转详情页的实现方法
2017/07/12 jQuery
Vue响应式原理深入解析及注意事项
2017/12/11 Javascript
看看“疫苗查询”小程序有温度的代码
2018/07/31 Javascript
浅谈React之状态(State)
2018/09/19 Javascript
微信小程序实现留言功能
2018/10/31 Javascript
JavaScript使用canvas绘制随机验证码
2020/02/17 Javascript
Python 执行字符串表达式函数(eval exec execfile)
2014/08/11 Python
使用python开发vim插件及心得分享
2014/11/04 Python
python根据出生年份简单计算生肖的方法
2015/03/27 Python
Python实现按中文排序的方法示例
2018/04/25 Python
python快排算法详解
2019/03/04 Python
python实现月食效果实例代码
2019/06/18 Python
python set集合使用方法解析
2019/11/05 Python
简单了解Pandas缺失值处理方法
2019/11/16 Python
django xadmin中form_layout添加字段显示方式
2020/03/30 Python
详解Django中异步任务之django-celery
2020/11/05 Python
购买美国制造的相框和画框架:Picture Frames
2018/08/14 全球购物
递归实现回文判断(如:abcdedbca就是回文,判断一个面试者对递归理解的简单程序)
2013/04/28 面试题
英语自荐信范文
2013/12/11 职场文书
2014年乡镇植树节活动方案
2014/02/28 职场文书
三八节标语
2014/06/27 职场文书
竞选学委演讲稿
2014/09/13 职场文书
小学少先队工作总结2015
2015/05/26 职场文书
幼儿园国培研修日志
2015/11/13 职场文书
2016高三毕业赠言寄语
2015/12/04 职场文书
医务人员医德医风心得体会
2016/01/25 职场文书
原生JS实现飞机大战小游戏
2021/06/09 Javascript