I Try Do

函数去抖动之debounce

snippet-key @js function debounce

underscore v1.7.0相关的源码剖

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
// underscore v1.7.0相关的源码剖
_.debounce = function(func, wait, immediate) {
// immediate默认为false
var timeout, args, context, timestamp, result;

var later = function() {
// 当wait指定的时间间隔期间多次调用_.debounce返回的函数,则会不断更新timestamp的值,导致last < wait && last >= 0一直为true,从而不断启动新的计时器延时执行func
var last = _.now() - timestamp;

if (last < wait && last >= 0) {
timeout = setTimeout(later, wait - last);
} else {
timeout = null;
if (!immediate) {
result = func.apply(context, args);
if (!timeout) context = args = null;
}
}
};

return function() {
context = this;
args = arguments;
timestamp = _.now();
// 第一次调用该方法时,且immediate为true,则调用func函数
var callNow = immediate && !timeout;
// 在wait指定的时间间隔内首次调用该方法,则启动计时器定时调用func函数
if (!timeout) timeout = setTimeout(later, wait);
if (callNow) {
result = func.apply(context, args);
context = args = null;
}

return result;
};
};

个人实现

功能和underscore实现一样, 去除了underscore的依赖

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
  /**
* 包裹函数 一段时间间隔只执行函数一次
* @param funcRef 函数引用
* @param wait default, 300ms
* 是否立即触发,触发后wait时间内不触发该方法,wait时间后才可以重新触发
* false:wait后触发 没有返回值
* true: wait前触发, 可以有返回值
* @param immediate default false
* return : 返回新的函数,使用:将需要调用funcRef的地方,替换为调用新返回的函数即可
*/
function debounce(funcRef, wait, immediate) {
var context, args, timestamp, timeout;
wait = wait || 300;
return function () {
context = this;
args = arguments;
timestamp = new Date().getTime();
var nowCall = immediate && !timeout;
if(!timeout){
timeout = setTimeout(later, wait)
}
if(nowCall){
var result = funcRef.apply(context, arguments);
context = args = null;
return result;
}
};
function later() {
// 当wait指定的时间间隔期间多次调用_.debounce返回的函数,则会不断更新timestamp的值,导致last < wait && last >= 0一直为true,从而不断启动新的计时器延时执行func
var last = new Date().getTime() - timestamp;
if (last < wait && last >= 0) {
timeout = setTimeout(later, wait - last);
}else {
timeout = null;
if (!immediate) {
funcRef.apply(context, args);
if (!timeout) context = args = null;
}
}
}
}