刚学习javascript的时候,感觉setTimeout很好理解,不就是过n(传入的毫秒数)毫秒,执行以下传入的函数吗?这个理解伴随了我挺长的一段时间,才对setTimeout有了新的认识,请先看下面的例子:

var start = new Date()

setTimeout(function(){

    var end = new Date()

    console.log("时间间隔:", end - start, "ms")

}, 500)

while( new Date() - start < 1000 ){}

如果是刚开始学习javascript的我可能会得到500ms的结果,真实的结果大约是一个1010ms的结果,为什么会这样那?

要能清楚这个问题,要先知道浏览器是怎样运行javascript的。在大多数浏览器中,执行的javascript代码和用户的UI界面更新是共用一个线程的,它的工作原理是这样的:与线程对应着一个简单的队列,每当执行的javascript代码或更新用户的UI界面时,处理任务会先进入等待队列,当线程空闲时,最先进入队列的任务就会被提取出来运行。如果我现在点击一个网页中的 button按钮,并且点击按钮触发一个click事件。代码如下:

document.getElementById("btn").onclick = function(){
dosomething()
var div = document.createElement("div")
div.innerHTML = "test"
document.body.appendChild(div)
}

浏览器对刚才操作的处理大致如下图:

当点击button按钮时,浏览器会创建两个任务加入到线程的队列中。第一个任务是更新按钮的样式,让用户知道按钮被点击了;第二个任务是执行click事件触发对应的javascript代码。假设这个时候线程是空闲状态,那么第一个任务就会被提取出来并执行,然后第二个任务就会被提取出来运行,在执行过程中,javascript代码创建了一个新的div元素,并追加到body元素的后面,这其实引发了另一次UI的变化。这意味着,在javascript代码运行过程中,一个新的UI更新任务被加入到队列中,当第二个任务完成后,UI还会再更新一次。

知道了UI线程的工作原理,再回头看上面的例子就不难理解,函数不是在500ms的时候立即执行,而是在500ms的时候加入等待队列。加入等待队列后发现现在线程不是空闲的,因为while在500ms的时候还在执行,当while执行结束后,也就是1010ms左右以后,线程才空闲。所以最后的结果是1010ms左右。如图:

所以类似如下代码返回什么值就很好理解了

for(var i =1; i<=3; i++ ){
setTimeout(function(){
console.log( i )
},0)
}

与setTineout类似的还有setInterval,但是setInterval会重复添加javascript任务到队列。当队列中已存在同一个setInterval创建的任务时,后续任务就不会添加到队列中。基于setInterval的设计,会导致两个问题:一个是某些间隔会被跳过;一个是多个定时器的代码执行间隔会比预期的小。看一下下面这段代码:

document.getElementById("btn").onclick = function(){
var start = new Date()
console.log("开始...")
setInterval( function(){
var start = new Date()
console.log("interval begin")
while( new Date() - start < 3200 ){}
console.log( "interval end")
}, 2000 )
while( new Date() - start < 3200 ){}
}

点击一个button按钮,执行上面的代码,设置了一个200ms间隔的重复定时器,click事件处理程序大约运行了3200ms,定时器代码也大约运行了3200ms。分析如图:

第一个定时器是在2000ms左右加入到队列的,但是这个时候click事件处理程序还没有运行完,等到3200ms左右这个时刻,click事件处理程序运行完,在队列中的第一个定时器任务会进入线程运行。在4000ms左右的时间,第二个定时器任务会进入等待队列。这个时候第一个定时器代码还在运行,并且在6000ms左右的时间第一个定时器代码还是在运行的。这样第二个定时器任务还在等待队列中,所以在6000ms左右,添加不了第三个定时器任务到等待队列。更有问题的是当第一个定时器任务运行结束后,第二个定时器的代码会立即执行。

为了避免setInterval的问题,聪明的前人想到了用setTimeout模拟setInterval。上面的代码可以改写成:

document.getElementById("btn").onclick = function(){
var start = new Date()
console.log("开始...")
setTimeout( function(){
var start = new Date()
console.log("interval begin")
while( new Date() - start < 3200 ){}
console.log( "interval end")
setTimeout(arguments.callee, 2000)
}, 2000 )
while( new Date() - start < 3200 ){}
}

这样做的好处是在本次定时代码执行完以前,不会向等待队列中添加新的定时器。这样在下一次定时器代码执行之前,至少会等待指定的时间间隔。同时因为是setTimeout模拟也不会缺失。

setTimeout浅析的更多相关文章

  1. setTimeout 与 Event Loop 浅析

    先从一个小题目开始: 以下代码的输出结果是? function test1 () { console.log(1) }; setTimeout(test1, 1000); // T1-1setTime ...

  2. setTimeout与setInterval的区别浅析

    在网页制作动态效果时,一定会遇到某些需求,要求某段程序等待多时时间后再开始执行,就像在我们的生活中一样,待会儿再开始做一件事.在JavaScript中主要通过定时器实现此类需求,本文将对定时器做一个概 ...

  3. 【深入浅出jQuery】源码浅析2--奇技淫巧

    最近一直在研读 jQuery 源码,初看源码一头雾水毫无头绪,真正静下心来细看写的真是精妙,让你感叹代码之美. 其结构明晰,高内聚.低耦合,兼具优秀的性能与便利的扩展性,在浏览器的兼容性(功能缺陷.渐 ...

  4. 【深入浅出jQuery】源码浅析2--使用技巧

    最近一直在研读 jQuery 源码,初看源码一头雾水毫无头绪,真正静下心来细看写的真是精妙,让你感叹代码之美. 其结构明晰,高内聚.低耦合,兼具优秀的性能与便利的扩展性,在浏览器的兼容性(功能缺陷.渐 ...

  5. 浅析Node.js的Event Loop

    目录 浅析Node.js的Event Loop 引出问题 Node.js的基本架构 Libuv Event Loop Event Loop Phases Overview Poll Phase The ...

  6. Android WebView File域同源策略绕过漏洞浅析

       0x00     我们首先讲一个webView这种方法的作用: webView.getSettings().setAllowFileAccessFromFileURLs(false);     ...

  7. Nodejs 基础知识 浅析

    1. 模块化 ①常用模块化规范 CommonJS + nodejs AMD(Asynchronous Module Definition) + RequireJS CMD(Common Module ...

  8. 分布式锁----浅析redis实现

    引言大概两个月前小伙伴问我有没有基于redis实现过分布式锁,之前看redis的时候知道有一个RedLock算法可以实现分布式锁,我接触的分布式项目要么是github上开源学习的,要么是小伙伴们公司项 ...

  9. Vue.nextTick浅析

    Vue.nextTick浅析 Vue的特点之一就是响应式,但数据更新时,DOM并不会立即更新.当我们有一个业务场景,需要在DOM更新之后再执行一段代码时,可以借助nextTick实现.以下是来自官方文 ...

随机推荐

  1. Commons-logging + Log4j

    一.Commons-logging能帮我们做什么? 1.提供一个统一的日志接口,简单了操作,同时避免项目与某个日志实现系统紧密耦合 2.自动选择适当的日志实现系统 a.classpath下查找comm ...

  2. TIMAC 学习笔记(二)

    昨天大体上熟悉了TIMAC自带的CC2530的示范例程,今天先从演示抓包入手,分析四种不同的配置工程在空中传输的差异.随后,会按照扫描.组网.入网等MAC层接口函数入手,结合IEEE 802.15.4 ...

  3. Windows Phone使用sliverlight toolkit实现页面切换动画效果

    使用应用时,好多app在页面切换的时候都有一个动画效果,感觉很炫,也大大增加了用户体验,怎么实现呢? 界面的切换,可以用Windows Phone Toolkit中的TransitionService ...

  4. 声明(创建) JavaScript 变量

    在 JavaScript 中创建变量通常称为"声明"变量. 我们使用 var 关键词来声明变量: var carname; 变量声明之后,该变量是空的(它没有值). 如需向变量赋值 ...

  5. 10套免费的响应式布局 Bootstrap 模版

    1. Cardio Cardio是我最喜欢的一个轻量级模板.它几乎可以很少的修改的用于任何类型的业务. 2. Evento Evento 是一个事件引导广告模板的形状.它是设计精美和注意细节. 3. ...

  6. Asp.net Response.Redirect with post data

    string url = String.Format("{0}://{1}/{2}", Request.Url.Scheme, Request.Url.Authority, &qu ...

  7. JAVA对象是如何占用内存的

      本文使用的是32位的JVM ,jdk1.6.本文基本是翻译的,加上了一些自己的理解,原文见文章底下链接.     在本文中,我们讨论如何计算或者估计一个JAVA对象占多少内存空间.(注意,使用 C ...

  8. RHEL 7.2 安装Oracle XE-11.2.0

    轻量快捷版本,适合开发 0. /etc/hosts 添加 本机hostname # hostnamepromote.cache-dns.local # cat /etc/hosts127.0.0.1 ...

  9. 国产CPU研究单位及现状

    1.国产CPU主要研制单位 (1)高性能通用CPU(“大CPU”,主要应用于高性能计算及服务器等) 主要研发单位:中国科学院计算所.北大众志.国防科技大学.上海高性能集成电路设计中心 (2)安全适用计 ...

  10. 可变参数列表---以dbg()为例

    在UART驱动的drivers/serial/samsung.h中遇到如下定义: #ifdef CONFIG_SERIAL_SAMSUNG_DEBUG extern void printascii(c ...