开发者社区> 技术小美> 正文
阿里云
为了无法计算的价值
打开APP
阿里云APP内打开

JavaScript定时器分析

简介:
+关注继续查看

一、事件循环

JavaScript是单线程,同一个时间只能做一件事情,所以执行任务需要排队。如果前一个耗时很长,那么下一个只能等待。

1)两种任务

为了更好的处理任务,JavaScript语言的设计者将任务分为两种:同步任务(synchronous)与异步任务(asynchronous)。

同步任务:在主线程上排队执行的任务。

异步任务:放在“任务队列”(task queue)中,只有当主线程空了,才会将“任务队列”中的任务放到主线程中。

这就是JavaScript的运行机制,这个过程会不断重复,这个机制叫事件循环(Event Loop)。

2)事件循环

事件循环模型可以用下图描述,图片来自Philip Roberts的演讲《Help, I’m stuck in an event loop》:

1. “WebAPIs”内的就是异步任务,包括DOM事件、Ajax和setTimeout。

2. “callback queue”内的是一个任务队列,包括click、load、done。

3. “stack”内的就是同步任务,只有当“stack”内的清空后,才会去轮询任务队列。

下面是一段代码说明,图片中的内容是打印结果,没什么悬念。

console.log('Hi');
setTimeout(function() {
  console.log('there');
},5000);
console.log('SJS');

1. 将log('Hi')方法入栈,这是个普通方法,出栈被引擎执行,输出“Hi”。

2. 将setTimeout方法入栈,这是个WebAPIs内的方法,出栈被引擎交给了相应模块,继续处理后面代码。

3. 将log('SJS')加入执行栈,出栈输出“SJS”。

4. 在setTimeout方法执行5秒后,到达触发条件,将setTimeout加到任务队列中。

5. 此时的执行栈为空,所以引擎开始轮询检查任务队列,有个setTimeout,于是将setTimeout加入执行栈中。

6. 在setTimeout中有个log('there')方法,将此方法入栈,输出“there”。

3)循环过程去取Ajax

下图展示的是主线程通过事件循环过程去取Ajax的消息:

 

二、定时器

定时器就是setTimeout(fn, delay)setInterval(fn, delay),定时器设定的延时是没有保证的。

如果setTimeout在时间点“n”被调用,那么执行定时器代码的JavaScript任务会在“n+delay”后才加入到消息队列中。

下图是JQuery的作者John Resig画的一张示例图:

左边是运行时间(单位ms),中间是JavaScript代码段,右边是代码计划开始执行时间

1. 0ms时JavaScript开始执行,2ms启动setTimeout,6ms加入Mouse Click,10ms启动setInterval,12ms加入setTimeout,20ms加入setInterval,30ms、40ms、50ms加入setInterval。

2. 第1段JavaScript执行了大概18ms,在18ms时,setTimeout过期了。

3. 按照单线程FIFO规则,接下来执行Mouse Click,再依次运行setTimeout和setInterval。

4. 第1个setInterval(20ms加入)还在排队等候中,30ms又要加入setInterval,但浏览器只让一个setInterval排队,其它的都废弃掉。

5. 第1个setInterval在36ms时开始执行,此程序需要执行6ms,第2个setInterval在40ms时开始排队,42ms时开始执行。

6. 第2个执行的setInterval在48ms时完成执行,50ms时第3个setInterval开始执行,不需要排队,直接运行。

 

三、分割任务

在JavaScript执行的时候,页面渲染的所有更新操作都要暂停。在执行繁忙的时候,可能会导致浏览器很卡或似乎要挂掉。

定时器,可以有效暂停一段JavaScript代码的执行,还可以将代码的各个部分,分解成不会让浏览器挂掉的碎片

1)未优化代码

长时间运行的任务,用手机扫二维码看效果会更明显

var tbody = document.getElementsByTagName("tbody")[0];
for (var i = 0; i < 20000; i++) { //创建20000个tr
  var tr = document.createElement("tr");
  for (var t = 0; t < 6; t++) { //每个tr有6个td
    var td = document.createElement("td");
    td.appendChild(document.createTextNode(i + "," + t));
    tr.appendChild(td);
  }
  tbody.appendChild(tr); //将tr添加到tbody中
}

2)已优化代码

利用定时器分解任务,将强循环转化为非阻塞操作:

//配置部分
var rowCount = 20000; 
var divideInto = 4; 
var chunkSize = rowCount / divideInto; //将20000分成4个部分
var iteration = 0; 

var table = document.getElementsByTagName("tbody")[0];

setTimeout(function generateRows() {
  var base = chunkSize * iteration; //计算上次中断的地方
  //添加tr部分
  for (var i = 0; i < chunkSize; i++) {
    var tr = document.createElement("tr");
    for (var t = 0; t < 6; t++) {
      var td = document.createElement("td");
      td.appendChild(document.createTextNode((i + base) + "," + t + "," + iteration));
      tr.appendChild(td);
    }
    table.appendChild(tr);
  }
  iteration++; //调度下一阶段
  if (iteration < divideInto) 
    setTimeout(generateRows, 0); 
}, 0);

 

 

参考资料:

JavaScript的计时器的工作原理

JavaScript:彻底理解同步、异步和事件循环(Event Loop)

从setTimeout说事件循环模型

JavaScript 运行机制详解:再谈Event Loop

JavaScript忍者秘籍

一家之言:说说 JavaScript 计时器的工作原理

分类: JavaScript


本文转自 咖啡机(K.F.J) 博客园博客,原文链接:http://www.cnblogs.com/strick/p/6644528.html   ,如需转载请自行联系原作者

版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

相关文章
Javascript定时器(三)——setTimeout(func, 0)
setTimeout(func, 0)可以使用在很多地方,拆分循环、模拟事件捕获、页面渲染等
39 0
剑指Offer——正则表达式匹配(JS实现)
剑指Offer——正则表达式匹配(JS实现)
65 0
ArcGIS JS 学习笔记4 实现地图联动
原文:ArcGIS JS 学习笔记4 实现地图联动 1.开篇       守望屁股实在太好玩了,所以最近有点懒,这次就先写个简单的来凑一下数。这次我的模仿目标是天地图的地图联动。 天地的地图联动不仅地图有联动,而且鼠标也有联动,我就照着这个目标进行山寨。
881 0
Arcgis for JS之Cluster聚类分析的实现
原文:Arcgis for JS之Cluster聚类分析的实现 在做项目的时候,碰见了这样一个问题:给地图上标注点对象,数据是从数据库来 的,包含XY坐标信息的,通过graphic和graphiclayer 的方式添加到地图上,其中有一个对象的数量很多,上万了吧,通过上述的方式无法在地图上进行展示,就想到了聚类,当时由于技术和时间的关系,没有实现,最 近,稍微有点先下时间,就又想起这事,继续研究,终于,皇天不负有心人,出来了,出来的第一时间写出来,以便大家使用。
1101 0
js template实现方法
培训类型:入园-家长会 待回访 ...
852 0
网站实现特定某个地区访问执行跳转(js方法)
DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.
814 0
asp.net WebService实现跨域js调用功能实现
1、Web.Config中增加如下红色标记部分配置:                                                               2、增加一...
815 0
ArcGIS JS 学习笔记1 用ArcGIS JS 实现仿百度地图的距离量测和面积量测
原文:ArcGIS JS 学习笔记1 用ArcGIS JS 实现仿百度地图的距离量测和面积量测 一、开篇      在博客注册了三年,今天才决定写第一篇博客,警告自己不要懒!!! 二、关于ArcGIS JS 版本选择      在写这篇博客时ArcGIS JS 4.0正式版已经发布。
1393 0
+关注
6819
文章
0
问答
文章排行榜
最热
最新
相关电子书
更多
低代码开发师(初级)实战教程
立即下载
阿里巴巴DevOps 最佳实践手册
立即下载
冬季实战营第三期:MySQL数据库进阶实战
立即下载