javascript异步处理工作机制详解


Posted in Javascript onApril 13, 2015

从基础的层面来讲,理解JavaScript的定时器是如何工作的是非常重要的。计时器的执行常常和我们的直观想象不同,那是因为JavaScript引擎是单线程的。我们先来认识一下下面三个函数是如何控制计时器的。

var id = setTimeout(fn, delay); - 初始化一个计时器,然后在指定的时间间隔后执行。该函数返回一个唯一的标志ID(Number类型),我们可以使用它来取消计时器。
var id = setInterval(fn, delay); - 和setTimeout有些类似,但它是连续调用一个函数(时间间隔是delay参数)直到它被取消。
clearInterval(id);, clearTimeout(id); - 使用计时器ID(setTimeout 和 setInterval的返回值)来取消计时器回调的发生
为了理解计时器的内在执行原理,有一个重要的概念需要加以探讨:计时器的延迟(delay)是无法得到保障的。由于所有JavaScript代码是在一个线程里执行的,所有异步事件(例如,鼠标点击和计时器)只有拥有执行机会时才会执行。

在这个图表中有许多信息需要理解,如果完全理解了它们,你会对JavaScript引擎如何实现异步事件有一个很好的认识。这是一个一维的图标:垂 直方向表示时间,蓝色的区块表示JavaScript代码执行块。例如第一个JavaScript代码执行块需要大约18ms,鼠标点击所触发的代码执行 块需要11ms,等等。

由于JavaScript引擎同一时间只执行一条代码(这是由于JavaScript单线程的性质),所以每一个JavaScript代码执行块会 “阻塞”其它异步事件的执行。这就意味着当一个异步事件发生(例如,鼠标点击,计时器被触发,或者Ajax异步请求)后,这些事件的回调函数将排在执行队 列的最后等待执行(实际上,排队的方式根据浏览器的不同而不同,所以这里只是一个简化);

从第一个JavaScript执行块开始研究,在第一个执行块中两个计时器被初始化:一个10ms的setTimeout()和一个10ms的setInterval()。 依据何时何地计时器被初始化(计时器初始化完毕后就会开始计时),计时器实际上会在第一个代码块执行完毕前被触发。但是,计时器上绑定的函数不会立即执行 (不被立即执行的原因是JavaScript是单线程的)。实际上,被延迟的函数将依次排在执行队列的最后,等待下一次恰当的时间再执行。

此外,在第一个JavaScript执行块中我们看到了一个“鼠标点击”事件发生了。一个JavaScript回调函数绑定在这个异步事件上了(我 们从来不知道用户什么时候执行这个(点击)事件,因此认为它是异步的),这个函数不会被立即执行,和上面的计时器一样,它将排在执行队列的最后,等待下一 次恰当的时候执行。

当第一个JavaScript执行块执行完毕后,浏览器会立即问一个问题:哪个函数(语句)在等待被执行?在这时,一个“鼠标点击事件处理函数”和 一个“计时器回调函数”都在等待执行。浏览器会选择一个(实际上选择了“鼠标点击事件的处理函数”,因为由图可知它是先进队的)立即执行。而“计时器回调 函数”将等待下次适合的时间执行。

注意,当“鼠标点击事件处理函数”执行的时候,setInterval的回调函数第一次被触发了。和setTimeout的回调函数一样,它将排到执行队列的最后等待执行。但是,一定要注意这一点:当setInterval回调函数第二次被触发时(此时setTimeout函数仍在执行)setInterval的第一次触发将被抛弃掉。当一个很长的代码块在执行时,可能把所有的setInterval回调函数都排在执行队列的后面,代码块执行完之后,结果便会是一大串的setInterval回调函数等待执行,并且这些函数之间没有间隔,直到全部完成。所以,浏览器倾向于的当没有更多interval的处理函数在排队时再将下一个处理函数排到队尾(这是由于间隔的问题)。

我们能够发现,当第三个setInterval回调函数被触发时,之前的setInterval回调函数仍在执行。这就说明了一个很重要的事实:setInterval不会考虑当前正在执行什么,而把所有的堵塞的函数排到队列尾部。这意味着两次setInterval回调函数之间的时间间隔会被牺牲掉(缩减)。

最后,当第二个setInterval回调函数执行完毕后,我们可以看到没有任何程序等待JavaScript引擎执行了。这就意味着浏览器现在在等待一个新的异步事件的发生。在50ms时一个新的setInterval回调函数再次被触发,这时,没有任何的执行块阻塞它的执行了。所以它会立刻被执行。

让我们用一个例子来阐明setTimeout和setInterval之间的区别:

setTimeout ( function ( ) { 
   /* Some long block of code... */ 
  setTimeout (arguments. callee, 10 ); 
  }, 10 ); 
  
 setInterval ( function ( ) { 
   /* Some long block of code... */ 
  }, 10 );

这两句代码乍一看没什么差别,但是它们是不同的。setTimeout回调函数的执行和上一次执行之间的间隔至少有10ms(可能会更多,但不会少于10ms),而setInterval的回调函数将尝试每隔10ms执行一次,不论上次是否执行完毕。

在这里我们学到了很多知识,总结一下:

JavaScript引擎是单线程的,强制所有的异步事件排队等待执行
setTimeout 和 setInterval 在执行异步代码的时候有着根本的不同
如果一个计时器被阻塞而不能立即执行,它将延迟执行直到下一次可能执行的时间点才被执行(比期望的时间间隔要长些)
如果setInterval回调函数的执行时间将足够长(比指定的时间间隔长),它们将连续执行并且彼此之间没有时间间隔。

以上所述就是本文的全部内容了,希望能够对大家学习javascript异步处理有所帮助。

Javascript 相关文章推荐
js或css文件后面跟参数的原因说明
Jan 09 Javascript
Array的push与unshift方法性能比较分析
Mar 05 Javascript
Dom操作之兼容技巧分享
Sep 20 Javascript
jquery ajaxSubmit 异步提交的简单实现
Feb 28 Javascript
javascript关于继承的用法汇总
Dec 20 Javascript
Vue.js 60分钟快速入门教程
Mar 28 Javascript
Node.js 8 中的 util.promisify的详解
Jun 12 Javascript
Angular4 ElementRef的应用
Feb 26 Javascript
小程序rich-text组件如何改变内部img图片样式的方法
May 22 Javascript
node 文件上传接口的转发的实现
Sep 23 Javascript
layui 解决富文本框form表单提交为空的问题
Oct 26 Javascript
vue中的v-model原理,与组件自定义v-model详解
Aug 04 Javascript
JavaScript中DOM详解
Apr 13 #Javascript
js 获取元素在页面上的偏移量的方法汇总
Apr 13 #Javascript
javascript中scrollTop详解
Apr 13 #Javascript
jQuery实现的在线答题功能
Apr 12 #Javascript
jQuery插件bxSlider实现响应式焦点图
Apr 12 #Javascript
jQuery插件Skippr实现焦点图幻灯片特效
Apr 12 #Javascript
jQuery插件pagination实现分页特效
Apr 12 #Javascript
You might like
单位速度在实战中的运用
2020/03/04 星际争霸
php 仿Comsenz安装效果代码打包提供下载
2010/05/09 PHP
基于php和mysql的简单的dao类实现crud操作功能
2014/01/27 PHP
php中filter_input函数用法分析
2014/11/15 PHP
用 Javascript 验证表单(form)中的单选(radio)值
2009/09/08 Javascript
ExtJS Grid使用SimpleStore、多选框的方法
2009/11/20 Javascript
JS将数字转换成三位逗号分隔的样式(示例代码)
2014/02/19 Javascript
简单的邮箱登陆的提示效果类似于yahoo邮箱
2014/02/26 Javascript
js 获取input点选按钮的值的方法
2014/04/14 Javascript
使用jQuery实现更改默认alert框体
2015/04/13 Javascript
jquery实现键盘左右翻页特效
2015/04/30 Javascript
javascript新闻跑马灯实例代码
2020/07/29 Javascript
Vue.js实现表格动态增加删除的方法(附源码下载)
2017/01/20 Javascript
vue-router中scrollBehavior的巧妙用法
2018/07/09 Javascript
详解redux异步操作实践
2018/08/15 Javascript
Node.js的进程管理的深入理解
2019/01/09 Javascript
vue实现Excel文件的上传与下载功能的两种方式
2019/06/28 Javascript
linux 下以二进制的方式安装 nodejs
2020/02/12 NodeJs
使用js和canvas实现时钟效果
2020/09/08 Javascript
[01:05:00]2018国际邀请赛 表演赛 Pain vs OpenAI
2018/08/24 DOTA
网红编程语言Python将纳入高考你怎么看?
2018/06/07 Python
对python3 中方法各种参数和返回值详解
2018/12/15 Python
Django如何自定义model创建数据库索引的顺序
2019/06/20 Python
Python Pivot table透视表使用方法解析
2020/09/11 Python
让IE支持CSS3的不完全兼容方案
2014/09/19 HTML / CSS
MAC彩妆英国官网:M·A·C UK
2018/05/30 全球购物
优瑞自动咖啡机官网:Jura
2018/09/29 全球购物
印尼在线旅游门户网站:NusaTrip
2019/11/01 全球购物
数控专业大学生的自我鉴定
2013/11/13 职场文书
门店业绩提升方案
2014/06/08 职场文书
财务工作检讨书
2014/10/29 职场文书
技术负责人岗位职责
2015/02/10 职场文书
违纪学生保证书
2015/02/27 职场文书
未婚证明范本
2015/06/15 职场文书
《称赞》教学反思
2016/02/17 职场文书
幼儿园小班教学反思
2016/03/03 职场文书