setTimeout和setInterval的区别你真的了解吗?


Posted in Javascript onMarch 31, 2011

甚至可能会错误的把两个实现定时调用的函数理解成了类似thread一样的东西, 认为会在一个时间片内, 并发的执行调用的函数, 似乎很好很强大, 但其实并不是如此, 实际的情况是javascript都是以单线程的方式运行于浏览器的javascript引擎中的, setTimeout和setInterval的作用只是把你要执行的代码在你设定的一个时间点插入js引擎维护的一个代码队列中, 插入代码队列并不意味着你的代码就会立马执行的,理解这一点很重要. 而且setTimeout和setInterval还有点不一样.

先谈谈setTimeout

function click() { 
// code block1... 
setTimeout(function() { 
// process ... 
}, 200); 
// code block2 
}

假设我们给一个button的onclick事件绑定了此方法, 当我们按下按钮后, 肯定先执行block1的内容, 然后运行到setTimeout的地方, setTimeout会告诉浏览器说, "200ms后我会插一段要执行的代码给你的队列中", 浏览器当然答应了(注意插入代码并不意味着立马执行), setTimeout代码运行后, 紧跟其后的block2代码开始执行, 这里就开始说明问题了, 如果block2的代码执行时间超过200ms, 那结果会是如何? 或许按照你之前的理解, 会理所当然的认为200ms一到, 你的process代码会立马执行...事实是, 在block2执行过程中(执行了200ms后)process代码被插入代码队列, 但一直要等click方法执行结束, 才会执行process代码段, 从代码队列上看process代码是在click后面的, 再加上js以单线程方式执行, 所以应该不难理解. 如果是另一种情况, block2代码执行的时间<200ms, setTimeout在200ms后将process代码插入到代码队列, 而那时执行线程可能已经处于空闲状态了(idle), 那结果就是200ms后, process代码插入队列就立马执行了, 就让你感觉200ms后, 就执行了.
再看看setInterval
这里可能会存在两个问题:
1.时间间隔或许会跳过
2.时间间隔可能<定时调用的代码的执行时间
function click() { 
// code block1... 
setInterval(function() { 
// process ... 
}, 200); 
// code block2 
}

和上面一样我们假设通过一个click, 触发了setInterval以实现每隔一个时间段执行process代码

setTimeout和setInterval的区别你真的了解吗?

比如onclick要300ms执行完, block1代码执行完, 在5ms时执行setInterval, 以此为一个时间点, 在205ms时插入process代码, click代码顺利结束, process代码开始执行(相当于图中的timer code), 然而process代码也执行了一个比较长的时间, 超过了接下来一个插入时间点405ms, 这样代码队列后又插入了一份process代码, process继续执行着, 而且超过了605ms这个插入时间点, 下面问题来, 可能你还会认为代码队列后面又会继续插入一份process代码...真实的情况是,由于代码队列中已经有了一份未执行的process代码, 所以605ms这个插入时间点将会被"无情"的跳过, 因为js引擎只允许有一份未执行的process代码, 说到这里不知道您是不是会豁然开朗呢...

为了这种情况你可以用一种更好的代码形式

setTimeout(function(){ 
//processing 
setTimeout(arguments.callee, interval); 
}, interval);

这个估计稍微想一下, 就明白其中的好处了, 这样就不会产生时间点被跳过的问题内容就到这里, 希望能有所帮助, 可能我表达的不是很清楚如果觉得自己英语基础不错可以直接看

setTimeout和setInterval的区别你真的了解吗?
里有关advanced Timers这节内容,  个人认为这本书真的很不错, 无论是想从零学起, 还是平日没事翻翻参考参考 都很不错, 作者是yahoo里很牛的一位前端开发工程师 : )

Javascript 相关文章推荐
JS 获取span标签中的值的代码 支持ie与firefox
Aug 24 Javascript
js导出table到excel同时兼容FF和IE示例
Sep 03 Javascript
HTML页面滚动时获取离页面顶部的距离2种实现方法
Sep 05 Javascript
jQuery实现类似滑动门切换效果的层切换
Sep 23 Javascript
AngularJS中的模块详解
Jan 29 Javascript
基于Jquery实现仿百度百科右侧导航代码附源码下载
Nov 27 Javascript
vuejs2.0子组件改变父组件的数据实例
May 10 Javascript
前端天气插件tpwidget使用方法详解
Jun 24 Javascript
jQuery内容选择器与表单选择器实例分析
Jun 28 jQuery
koa2 从入门到精通(小结)
Jul 23 Javascript
JavaScript使用百度ECharts插件绘制饼图操作示例
Nov 26 Javascript
详解Node.js如何处理ES6模块
May 15 Javascript
Draggable Elements 元素拖拽功能实现代码
Mar 30 #Javascript
使用jQuery实现dropdownlist的联动效果(sharepoint 2007)
Mar 30 #Javascript
使用jquery为table动态添加行的实现代码
Mar 30 #Javascript
jquery 查找select ,并触发事件的实现代码
Mar 30 #Javascript
lyhucSelect基于Jquery的Select数据联动插件
Mar 29 #Javascript
基于JQuery实现异步刷新的代码(转载)
Mar 29 #Javascript
使用隐藏的new来创建对象
Mar 29 #Javascript
You might like
从零开始 教你如何搭建Discuz!4.1论坛
2006/07/07 PHP
PHP利用REFERER根居访问来地址进行页面跳转
2013/09/28 PHP
smarty获得当前url的方法分享
2014/02/14 PHP
PHP include任意文件或URL介绍
2014/04/29 PHP
JavaScript监测ActiveX控件是否已经安装过的代码
2008/09/02 Javascript
JavaScript 核心参考教程 内置对象
2009/10/13 Javascript
js实现的真正的iframe高度自适应(兼容IE,FF,Opera)
2010/03/07 Javascript
jQuery学习总结之元素的相对定位和选择器(持续更新)
2011/04/26 Javascript
三种Node.js写文件的方式
2016/03/08 Javascript
jquery Deferred 快速解决异步回调的问题
2016/04/05 Javascript
Jquery修改image的src属性,图片不加载问题的解决方法
2016/05/17 Javascript
JS+Canvas 实现下雨下雪效果
2016/05/18 Javascript
canvas学习之API整理笔记(一)
2016/12/29 Javascript
COM组件中调用JavaScript函数详解及实例
2017/02/23 Javascript
js数字舍入误差以及解决方法(必看篇)
2017/02/28 Javascript
关于vue.js v-bind 的一些理解和思考
2017/06/06 Javascript
JavaScript取得gridview中获取checkbox选中的值
2017/07/24 Javascript
vue中使用vue-cli接入融云实现即时通信
2019/04/19 Javascript
vue页面加载时的进度条功能(实例代码)
2020/01/13 Javascript
vue组件中传值EventBus的使用及注意事项说明
2020/11/16 Javascript
[02:51]2018年度DOTA2最佳中单位选手-完美盛典
2018/12/17 DOTA
[04:20]DOTA2-DPC中国联赛 正赛 VG vs LBZS 选手采访 1月19日
2021/03/11 DOTA
详解python实现读取邮件数据并下载附件的实例
2017/08/03 Python
Python3多线程爬虫实例讲解代码
2018/01/05 Python
python调用虹软2.0第三版的具体使用
2019/02/22 Python
pytorch 模型可视化的例子
2019/08/17 Python
Python re正则表达式元字符分组()用法分享
2020/02/10 Python
Python如何解除一个装饰器
2020/08/07 Python
英国文胸专家:AmpleBosom.com
2018/02/06 全球购物
英国儿童鞋和靴子:Start-Rite
2019/05/06 全球购物
大学生党员自我批评
2014/02/14 职场文书
大学英语演讲稿范文
2014/04/24 职场文书
遗嘱继承权公证书
2015/01/26 职场文书
2015年工商所工作总结
2015/05/21 职场文书
宝宝满月祝酒词
2015/08/10 职场文书
golang DNS服务器的简单实现操作
2021/04/30 Golang