解析underscore中的throttle
什么是throttle(节流)
Throttling enforces a maximum number of times a function can be called over time.
简单来说就是你假设给定一个wait表示这在个时间内该函数最多可以被执行一次。我们知道知道浏览器scroll触发事件的频率非常高,如果不使用节流的话,我们轻轻一滚动鼠标滑轮可能就触发了10来次某个添加到scroll事件的函数。但如果我们使用节流这个技术的话,我们设置wait为1000(ms),当我们不停地滚动滑轮10s,函数最多被执行10次。10000 / 1000 = 10
最简单的节流
var throttle = function(func, wait){
var previous = 0;
return function(){
var now = +new Date();
if (now - previuos > wait){
func.apply(this, arguments);
last = now;
}
}
}
这个函数利用闭包返回一个函数,而且它有两个重要的特点:
- 当两次函数触发的时间间隔大于
wait时,func才会被调用 - 第一次触发时
func会被调用
underscore中的throttle
// Returns a function, that, when invoked, will only be triggered at most once
// during a given window of time. Normally, the throttled function will run
// as much as it can, without ever going more than once per `wait` duration;
// but if you'd like to disable the execution on the leading edge, pass
// `{leading: false}`. To disable execution on the trailing edge, ditto.
_.throttle = function(func, wait, options) {
var timeout, context, args, result;
var previous = 0;
if (!options) options = {};
var later = function() {
previous = options.leading === false ? 0 : _.now();
timeout = null;
result = func.apply(context, args);
if (!timeout) context = args = null;
};
var throttled = function() {
var now = _.now();
if (!previous && options.leading === false) previous = now;
var remaining = wait - (now - previous);
context = this;
args = arguments;
if (remaining <= 0 || remaining > wait) {
if (timeout) {
clearTimeout(timeout);
timeout = null;
}
previous = now;
result = func.apply(context, args);
if (!timeout) context = args = null;
} else if (!timeout && options.trailing !== false) {
timeout = setTimeout(later, remaining);
}
return result;
};
throttled.cancel = function() {
clearTimeout(timeout);
previous = 0;
timeout = context = args = null;
};
return throttled;
};
咋一看这个函数的实现比当初那个简单的函数长了很多, 别怕因为他们的思想是一模一样的,多余的代码只是为了一些额外的特性,并不复杂。
首先多了一个options参数,它是一个对象,可以设置leading和trailing属性。leading是提前领先的意思,在那个简单的版本中我们知道函数在第一次触发时候func是会被触发的,这就是leading。所以当我们没有设置{leading: false}时候,func会在第一次函数触发时候马上被执行。但是当我们显性地传入{leading: false}时候,func就不会马上执行。这是因为if (!previous && options.leading === false) previous = now; 开始previous为0那么条件均为真,previous = now即now - previous > wait不成立。
即第一次触发函数会进入到
else if (!timeout && options.trailing !== false) {
// var remaining = wait - (now - previous);
// now = previous;因此later会在wait毫秒后被执行
timeout = setTimeout(later, remaining);
}
再来看看later
var later = function() {
previous = options.leading === false ? 0 : _.now();
timeout = null;
result = func.apply(context, args);
if (!timeout) context = args = null;
};
其实写成这样更号理解
var later = function() {
previous = options.leading === false ? 0 : _.now();
// 为了让将previous设为0,是让if (!previous && options.leading === false)再次成立
// 意思就是当超过wait的时间没去触发函数了,再次触发时候的这次也算是首次,它不能马上被执行。(想象就是不断滑动滚轮10s,然后放下鼠标去喝口水,再回来滑滚轮,那应该算作新的一次开始,而不是上次的继续)
result = func.apply(context, args);
timeout = context = args = null;
};
但是如果第二次触发与第一次触发的时间间隔大于wait时候就会进入到
// 实际上remaining<=0就足够了,后者是考虑到假如客户端修改了系统时间则马上执行func函数
if (remaining <= 0 || remaining > wait) {
// 取消第一次的setTimeout
if (timeout) {
clearTimeout(timeout);
timeout = null;
}
previous = now;
result = func.apply(context, args);
if (!timeout) context = args = null;
}
其实也应该写成这样更好理解
if (remaining <= 0 || remaining > wait) {
if (timeout) {
clearTimeout(timeout);
}
previous = now;
result = func.apply(context, args);
timeout = context = args = null;
}
有个疑问就是imeout = setTimeout(later, remaining), remaining等于wait,如果两次时间间隔十分接近wait的又大于wait应该是怎么样的流程呢。个人觉得应该是进入到上面这个代码块然后clearTimeout, 为什么呢,首先javaScript是单线程的,setTimeout的意思是将函数在wait毫秒后添加到任务队列中,而不是立即执行。所以理论上来讲还是进入上述代码块要比在执行later()早。但是想想如果每次都是setTimeout也行,每隔wait运行later,效果差不多。
小结
所以第三个参数不传入就是leading模式。
{trailing: false}也是leading模式但和不传参数还是有点区别就是它无法执行timeout = setTimeout(later, remaining);。
{leading: false}就是trailing模式,他的timeout = setTimeout(later, remaining);实际上是timeout = setTimeout(later, wait)
解析underscore中的throttle的更多相关文章
- 解析underscore中的debounce
先奉上源码 取自Underscore.js 1.9.1的debounce _.debounce = function(func, wait, immediate) { var timeout, res ...
- 理解Underscore中的节流函数
上一篇中讲解了Underscore中的去抖函数(_.debounced),这一篇就来介绍节流函数(_.throttled). 经过上一篇文章,我相信很多人都已经了解了去抖和节流的概念.去抖,在一段连续 ...
- 理解Underscore中的_.bind函数
最近一直忙于实习以及毕业设计的事情,所以上周阅读源码之后本周就一直没有进展.今天在写完开题报告之后又抽空看了一眼Underscore源码,发现上次没有看明白的一个函数忽然就豁然开朗了,于是赶紧写下了这 ...
- 关于 underscore 中模板引擎的应用演示样例
//关于 underscore 中模板引擎的应用演示样例 <!doctype html> <html> <head> <meta charset=" ...
- 深入解析Underscore.js源码架构
Underscore.js是很有名的一个工具库,我也经常用他来处理对象,数组等,本文会深入解析Underscore源码架构,跟大家一起学习下他源码的亮点,然后模仿他写一个简单的架子来加深理解.他的源码 ...
- 浅解析js中的对象
浅解析js中的对象 原文网址:http://www.cnblogs.com/foodoir/p/5971686.html,转载请注明出处. 前面的话: 说到对象,我首先想到的是每到过年过节见长辈的时候 ...
- 深入解析Javascript中this关键字的使用
深入解析Javascript中面向对象编程中的this关键字 在Javascript中this关键字代表函数运行时,自动生成的一个内部对象,只能在函数内部使用.比如: function TestFun ...
- js中eval详解,用Js的eval解析JSON中的注意点
先来说eval的用法,内容比较简单,熟悉的可以跳过eval函数接收一个参数s,如果s不是字符串,则直接返回s.否则执行s语句.如果s语句执行结果是一个值,则返回此值,否则返回undefined. 需要 ...
- 2dx解析cocosbuilder中使用layer时的缺陷
2dx解析cocosbuilder中使用layer时的缺陷 cocos2d-x 3.7 cocosbuilder中的layer通常会用到触摸属性: 但是在2dx解析布局文件的时候,却很多属性都没解析: ...
随机推荐
- 1009 Product of Polynomials (25分) 多项式乘法
1009 Product of Polynomials (25分) This time, you are supposed to find A×B where A and B are two po ...
- Netsparker破解版5.3 Netsparker Enterprise 5.3.0.24388[cracked]
Netsparker破解版5.3 Netsparker Enterprise 5.3.0.24388[cracked]该版本更新时间为2019年7月8日下载地址:1 https://www.dr-fa ...
- mysql MVCC 实现原理
MVCC( Multi-Version Concurrency Controll) 每一行都存储了事件发生时的系统版本号(System Version Number),用来替代事件实际发生的时间.每一 ...
- android开发如何在页面之间传参
第一个页面跳转 传递值 Button bn1=(Button)findViewById(R.id.btn_Login); //跳转bn1.setOnClickListener(new OnClickL ...
- 跳转连接转base64
//配置H5连接 @Value("${h5.domain}") private String h5Domain; // 校验商家端权限 public String checkSta ...
- SRS源码——Listener
1. 整理了一下Listener相关的UML类图:
- WAV格式解析
WAV为微软公司(Microsoft)开发的一种声音文件格式,它符合RIFF(Resource Interchange File Format)文件规范,用于保存Windows平台的音频信息资源,被W ...
- C语言程序设计-现代方法(笔记3)
第十三章 字符串 1.字符串字面量(13.1) 字符串字面量:用一对双引号括起来的字符序列.字符串字面量可以像字符常量一样包含转义字序列. 在字符串字面量中小心使用八进制和十六进制的转义序列. 字符串 ...
- python 爬虫原理
简单来说互联网是由一个个站点和网络组成的大网,我们通过浏览器访问站点,站点把HTML.JS.CSS代码返回给浏览器,这些代码经过浏览器解析.渲染,将丰富多彩的网页呈现我们眼前: 一.爬虫是什么? 如果 ...
- C#委托-事件示例
Program.cs using System;using System.Collections.Generic;using System.Linq;using System.Text;using S ...