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:彻底理解同步、异步和事件循环(Event Loop)
JavaScript 运行机制详解:再谈Event Loop
JavaScript定时器分析的更多相关文章
- 一些有用的javascript实例分析(三)
原文:一些有用的javascript实例分析(三) 10 输入两个数字,比较大小 window.onload = function () { var aInput = document.getElem ...
- 关于JavaScript定时器我的一些小理解
因为自己在平时工作中,有些功能需要用到定时器,但是定时器并不像我们表边上看到的那样,所以这周末我看看书查查资料,深入研究了一下JavaScript中的定时器,那么废话不多说,下面进入我们今天的正题. ...
- Javascript 定时器的使用陷阱 (setInterval)
setTimeout(function(){ // 其他代码 setTimeout(arguments.callee, interval); }, interval); setInterval会产生回 ...
- Javascript定时器(三)——setTimeout(func, 0)
setTimeout(func, 0)可以使用在很多地方,拆分循环.模拟事件捕获.页面渲染等 一.setTimeout中的delay参数为0,并不是指马上执行 <script type=&quo ...
- Javascript定时器(二)——setTimeout与setInterval
一.解释说明 1.概述 setTimeout:在指定的延迟时间之后调用一个函数或者执行一个代码片段 setInterval:周期性地调用一个函数(function)或者执行一段代码. 2.语法 set ...
- [转载]JavaScript内存分析
https://github.com/CN-Chrome-DevTools/CN-Chrome-DevTools/blob/master/md/Performance-Profiling/javasc ...
- JavaScript 性能分析新工具 OneProfile
OneProfile 是一个网页版的小工具,可以用全新的方式展示 JavaScript 性能分析的结果,帮助开发者洞悉函数调用关系,优化应用性能. 点击打开 OneProfile 背景 Chrome ...
- 一些有用的javascript实例分析(一)
原文:一些有用的javascript实例分析(一) 本文以http://fgm.cc/learn/链接的实例索引为基础,可参见其实际效果.分析和整理了一些有用的javascript实例,相信对一些初学 ...
- 一些有用的javascript实例分析(二)
原文:一些有用的javascript实例分析(二) 5 求出数组中所有数字的和 window.onload = function () { var oBtn = document.getElement ...
随机推荐
- MySQL · 引擎特性 · InnoDB IO子系统
前言 InnoDB做为一款成熟的跨平台数据库引擎,其实现了一套高效易用的IO接口,包括同步异步IO,IO合并等.本文简单介绍一下其内部实现,主要的代码集中在os0file.cc这个文件中.本文的分析默 ...
- Laravel的console使用方法
适用场景:分析数据(日志) php artisan make:console 你的命令类名 示例: php artisan make:console Check 在\app\Console\Comma ...
- Scalatra--Introduction And Quick start
Introduction Scalatra是一款轻易级Scala web框架,通过Scalatra可以很轻易创建web Application,由Linkedln开源并遵循了Ruby Web框架的Si ...
- Android网络开发实例(基于抓包实现的网络模拟登录,登出和强制登出)
学习Android有几个月了,最近喜欢上了网络编程,于是想通过Android写一些一个小程序用于连接外网.在这里非常感谢雪夜圣诞的支持,非常感谢,给我打开新的一扇门. 1.声明,本程序只能用于西南大学 ...
- Python 之 json 模块
引言 对于做web开发的人来说,json文本必须要熟知与熟练使用的.大部分网站的API接口调用返回的数据,就是json格式的.如果看json对象所包含的内容,相信对熟悉Python的人开说,很快就能把 ...
- vue入门 vue与react和Angular的关系和区别
一.为什么学习vue.js vue.js兼具angular.js和react的优点,并且剔除了他们的缺点 官网:http://cn.vuejs.org/ 手册:http://cn.vuejs.org/ ...
- KoaHub.js -- 基于 Koa.js 平台的 Node.js web 快速开发框架之koahub-body-res
koahub body res Format koa's respond json. Installation $ npm install koahub-body-res Use with koa v ...
- 某电商网站线上drbd+heartbeat+nfs配置
1.环境 nfs1.test.com 10.1.1.1 nfs2.test.com 10.1.1.2 2.drbd配置 安装drbd yum -y install gcc gcc-c++ make g ...
- 道路修建 2(自创题+题解)(From NOI2011)
道路修建这道题想来各位不陌生(传送门在此——Bzoj2435),看了此题,一开始以为是最初各个点处于分散状态,然后做了一下,直到发现标程都有点问题,才发现原题是说本来各点已经处于连接完毕的状态(phi ...
- dfs 无向图两节点间的所有路径
标题:风险度量 X星系的的防卫体系包含 n 个空间站.这 n 个空间站间有 m 条通信链路,构成通信网.两个空间站间可能直接通信,也可能通过其它空间站中转. 对于两个站点x和y (x != y), 如 ...