1、什么是js函数节流

    其本质就是不让某些代码(函数)在没有间断的情况下连续重复执行,目的就是让执行函数的请求停止了一段时间后才执行。

  2、函数节流运用的场景

    窗口大小的改变(resize事件),滚动事件(scroll事件),鼠标移动事件(mousemove事件),touchmove事件(做过手机端的同学一定知道,手机中手指滑动其实触发了多次touchmove),我们在绑定这些事件的时候,函数会被多次执行,因为性能的需要,此时函数节流就派上用场!

  3、函数节流的书写基本形式

    网上也提供很多关于函数节流的模式,根据自身的需要,正常写法有以下几种:

    (1)对象字面量的方式:

<script type="text/javascript">
var throttle={
timeoutId:null,
performProcessing:function(){
console.log("节流")
},
process:function(){
clearTimeout(this.timeoutId);
var that=this;//为什么要设置这个,亲,setTimeout可是会改变this的指针指向window哦
this.timeoutId=setTimeout(function(){that.performProcessing()},250)
}
}
window.onresize=function(){
throttle.process();
}
</script>

  当调用process函数的时候,就清除timeoutId来阻止之前的队列函数,然后创建一个新的定时器去调用performProcessing。时间的间隔是250ms,指的是最后一次调用process()之后至少250ms后才会继续调用performProcessing。250ms,代表着在250ms的时间内,不管调用了多少次,performProcessing只会被执行一次(而且是在最后一次被调用的那个函数间隔250ms才添加到队列并执行,如果你一直不间断触发函数,那么performProcessing将永远不会被执行)。

  (2)函数式的方式:

function throttle(method,context){
clearTimeout(method.tId);
method.tId=setTimeout(function(){
method.call(context) //改变作用域
},250)
}
function resizeFunc(){
console.log(1)
}
window.onresize=function(){
throttle(resizeFunc)
}

  这种函数节流看来已经不错了,可是发现我们在调用的时候和第1种方式(对象字面量方式)是一样的,这个时候问题就来了。如果页面多次调用,显然这种方式是无法通过参数来改变函数执行的频率(delay),所以难形成共用!怎么办呢?

  那是否有更好的方法呢?如果能把delay的参数放出来呢?所以有了下面的函数写法:

<script type="text/javascript">
var throttle = function(fn, delay){
var timer = null;
return function(){var context = this, args = arguments;
clearTimeout(timer);
timer = setTimeout(function(){
fn.apply(context, args);
}, delay);
};
};
function printed(){
console.log(1)
}
window.onresize=throttle(printed,250)
</script>

  这种方式,脑海中想到的是闭包,没错!就是通过js的闭包的方式,返回函数,通过timer的变量来控制函数的执行!改造完成以后基本能够满足了我们的需求!感觉大功告成了,哈哈!

  可是,发现了没有,如果我不间断地执行某个函数,假设执行了300ms后才停下来,那么意味着setTimeout只有在300ms后才添加到队列,然后执行,那我们所想象的mousemove不就是一闪一闪的结果吗?(因为执行的频率太多频繁而无法立即出发setTimeout),所以我们肯定有必要在规定的时间内,移动要执行某个函数一次,这样就可以必选视觉的误差!所以我们规定了某个函数在视觉允许的范围内,固定时间至少调用一次!

<script type="text/javascript">
function throttle(fn,delay,duration){
var timer=null,last_exec=0;
return function(){
var context=this,args=arguments,elapsed = +new Date();
clearTimeout(timer);
if(!last_exec){
last_exec = elapsed;
}
if(elapsed-last_exec > duration){
fn.apply(context, args);
last_exec=elapsed;
}else {
timer = setTimeout(function(){
fn.apply(context, args);
}, delay);
}
}
} function resizeFunc(){
console.log(1)
}
window.onresize=throttle(resizeFunc,50,5000)
</script>

  函数的原理很简单,每次执行函数的时候,判断当前的时间elapsed与上一次函数执行的时间的间隔大于delay的时候(elapsed-last_exec)就执行函数fn,然后将last_exec设为当前的日期,这样就可以达到了我们的目的了!

  到这里,认为节流的函数已经结束了饿,可是你看的越多发现自己越无知,有木有呢?因为jquery最近版本已经有了throttle的函数了,把其中的代码如下:

(function(window,undefined){
'$:nomunge';
var $ = window.jQuery || window.Cowboy || ( window.Cowboy = {} ),jq_throttle;
$.throttle = jq_throttle = function( delay, no_trailing, callback, debounce_mode ) {
var timeout_id,last_exec = 0;
if ( typeof no_trailing !== 'boolean' ) {
debounce_mode = callback;
callback = no_trailing;
no_trailing = undefined;
}
function wrapper() {
var that = this,
elapsed = +new Date() - last_exec,
args = arguments;
function exec() {
last_exec = +new Date();
callback.apply( that, args );
};
function clear() {
timeout_id = undefined;
}; if ( debounce_mode && !timeout_id ) {
exec();
}
timeout_id && clearTimeout( timeout_id );
if ( debounce_mode === undefined && elapsed > delay ) {
exec();
} else if ( no_trailing !== true ) {
timeout_id = setTimeout( debounce_mode ? clear : exec, debounce_mode === undefined ? delay - elapsed : delay );
}
};
if ( $.guid ) {
wrapper.guid = callback.guid = callback.guid || $.guid++;
}
return wrapper;
};
$.debounce = function( delay, at_begin, callback ) {
return callback === undefined
? jq_throttle( delay, at_begin, false )
: jq_throttle( delay, callback, at_begin !== false );
}; })(this);

  throttle函数的几个参数的说明:

    delay:延迟的时间(Number)

    no_trailing:默认是false(Boolean),可选参数!如果是false,那么固定每个一段时间(delay)定会调用一次callBack函数,如果是true,那么意味着callBack只会执行最后一次在throttle函数被触发后(听起来晕晕的!请看下图):其实可以理解为:在250ms的时间内,如果函数200ms就执行完了,那么no_trailing=true的时候就不触发callBack,no_trailing=false的时候就促发callBack

  // > Throttled with `no_trailing` specified as false or unspecified:
// > ||||||||||||||||||||||||| (pause) |||||||||||||||||||||||||
// > X X X X X X X X X X X X
// >
// > Throttled with `no_trailing` specified as true:
// > ||||||||||||||||||||||||| (pause) |||||||||||||||||||||||||
// > X X X X X X X X X X

    callBack:就是间隔delay时间要执行的函数

    debounce_mode:可选,不传递任何参数就是undefined!如果是传递的是false的时候,那么就是追后一次执行事件delay的时间后调用callBack(没有固定的时间调用一次),如果debounce_mode=true,事件触发后立刻执行callBack,最后一次事件在delay的时间后触发callBack,开始就触发callBack函数(debounce_mode=false则不是)

  debounce函数的几个参数的说明:

    delay:延迟的时间(Number)

    at_begin:默认false,at_begin=false意味着callBack会在debounce函数最后一次触发函数后执行!

   // > Debounced with `at_begin` specified as false or unspecified:
// > ||||||||||||||||||||||||| (pause) |||||||||||||||||||||||||
// > X X
// >
// > Debounced with `at_begin` specified as true:
// > ||||||||||||||||||||||||| (pause) |||||||||||||||||||||||||
// > X X

    callBack:就是间隔delay时间要执行的函数。

    到这里,函数节流就算结束了,jquery中的封装比较难理解!但是我们只要知道调用方式和参数的作用即可!最后附上参考的网址:

  地址:http://benalman.com/projects/jquery-throttle-debounce-plugin/

												

也谈js函数节流的更多相关文章

  1. 浅谈JS函数节流及应用场景

    说完防抖,下面我们讲讲节流,规矩就不说了,先上代码: <!DOCTYPE html> <html lang="en"> <head> <m ...

  2. 浅谈javascript函数节流

    浅谈javascript函数节流 什么是函数节流? 函数节流简单的来说就是不想让该函数在很短的时间内连续被调用,比如我们最常见的是窗口缩放的时候,经常会执行一些其他的操作函数,比如发一个ajax请求等 ...

  3. js 函数节流和防抖

    js 函数节流和防抖 throttle 节流 事件触发到结束后只执行一次. 应用场景 触发mousemove事件的时候, 如鼠标移动. 触发keyup事件的情况, 如搜索. 触发scroll事件的时候 ...

  4. Underscore.js 函数节流简单测试

    函数节流在日常的DOM界面数据交互中有比较大的作用,可以减少服务器的请求,同时减少客户端的内存影响 Underscore.js  本省就包含了函数节流的处理函数 _.throttle 和 _.debo ...

  5. 结构-行为-样式-Js函数节流

    最近一个面试官问了我一个函数节流的问题,然后感觉自己工作中遇到过这个问题,但是不知道这种形式就是函数节流.下面我来说下这个Js的高级问题,思路:函数节流就是防止用户高频调用某个事件而做的Js层面的处理 ...

  6. js 函数节流throttle 函数去抖debounce

    1.函数节流throttle 通俗解释: 假设你正在乘电梯上楼,当电梯门关闭之前发现有人也要乘电梯,礼貌起见,你会按下开门开关,然后等他进电梯: 但是,你是个没耐心的人,你最多只会等待电梯停留一分钟: ...

  7. 深入理解JS函数节流和去抖动

    一.什么是节流和去抖? 1.节流 节流就是拧紧水龙头让水少流一点,但是不是不让水流了.想象一下在现实生活中有时候我们需要接一桶水,接水的同时不想一直站在那等着,可能要离开一会去干一点别的事请,让水差不 ...

  8. JS函数节流和防抖

    看JS高级程序设计时,了解到一个概念--函数节流,是为了防止在高频率触发某些事件导致浏览器崩溃.最近又了解到另一个概念,防抖,感觉和函数节流很像,也查看了很多篇博文,算是理解了. 区别: 函数节流:频 ...

  9. JS函数节流代码实现

    函数被频繁调用场景 Js中的函数大多数情况下都是由用户主动调用触发的,一般不会遇到性能相关的问题.但在一些少数情况下,函数的触发不是由用户直接控制.在这些场景下,函数有可能被非常频繁地调用,而造成大的 ...

随机推荐

  1. 终极解法According to TLD or attribute directive in tag file, attribute select does not accept any expressions

    3天硬是是把这个问题解决了 有时候突然上个厕所灵感就来了 第一次向用JSTL解析xml 然后我想遍历整个xml文档打印出来 居然不让我输入变量 那让我怎么办啊 在网上各种找答案 说什么<%@ t ...

  2. 手把手教你使用Git(转)

    Git使用教程 2014-10-25 14:29 by 云溪0707, 10532 阅读, ... 评论, 收藏, 编辑 Git使用教程 一:Git是什么? Git是目前世界上最先进的分布式版本控制系 ...

  3. 安装TDM-GCC

    TDM-GCC是一组免费的编译器套件,有32位和64位两种版本.其中64位版既可以编译生成64位的可执行文件,又可以编译生成32位的可执行文件.从TDM-GCC的官网可以下载到相应的安装包,安装完成后 ...

  4. javascript模拟html title

    这只是个学习原生javascript过程中的一个练手效果 不做说明解释 直接上代码 算是个参考 <!doctype html> <html> <head> < ...

  5. java学习:AWT组件和事件处理的笔记(1)--菜单条,菜单,菜单项

    菜单放在菜单条里,菜单项放在菜单里1.MenuBar    在java.awt包中,负责创建菜单条,即MenuBar的一个实例,便是一个菜单条.    在Frame类中的setMenuBar(Menu ...

  6. Qt5-MSVC2012-qDebug中文乱码解决方法

    1.首先看代码: int main(int argc, char *argv[]) { QApplication a(argc, argv); MainWindow w; w.show(); qDeb ...

  7. 转:C4项目中验证用户登录一个特性就搞定

    转:C4项目中验证用户登录一个特性就搞定   在开发过程中,需要用户登陆才能访问指定的页面这种功能,微软已经提供了这个特性.     // 摘要:    //     表示一个特性,该特性用于限制调用 ...

  8. python bool值要注意的一些地方

    1.像(),[],{}这三个是可以通过bool(()),bool([]),bool({})转化为bool值的:且它们转化后的结果为False.但是这三个值它本身并不等于False.切记不可以与Fals ...

  9. MYSQL 加密的 3 类方法

    背景: 若你想要储存一些由可能包含任意字节值的加密函数返回的结果,使用BLOB列而不是 CHAR 或VARCHAR 列,从而避免由于结尾空格的删除而改变一些数据值的潜在问题. 这一句话来自官方文件,记 ...

  10. 构建混合云:配置Azure site to site VPN连接(3)

    9. 那么我们来创建网关,创建网关的时候需要注意,看看你的设备是否支持动态网关,在本示例中的Cisco ASA 5550不支持动态网关,所以我们只能创建静态网关: 该创建会花费一定的时间,稍等即可. ...