JavaScript引擎是单线程运行的,浏览器无论在什么时候都只且只有一个线程在运行JavaScript程序.
setInterval 和 setTimeout ,都涉及到时间计数器,也就是都涉及到一个类似与MFC定时器。JS引擎本身就只能单线程运行,因此定时器需要由其他的外部线程来启动。所以对JS引擎而言,定时器线程可以被视为异步线程。但当定时器时间到达后,所触发的事件则必须在任务列表中排队,等候JS引擎的处理。
eg:
setTimeout( function(){ while(true){} } , 100);
setTimeout( function(){ alert(’你好!’); } , 200);
setInterval( callbackFunction , 200);
第一行代码进入死循环,你会发现后面两行代码都没有执行。这说明定时器并不是多线程异步执行的。
那么单线程的JavaScript引擎是怎么配合浏览器内核处理这些定时器和响应浏览器事件的呢?
浏览器中的线程介绍:
浏览器内核实现允许多个线程异步执行,这些线程在内核制控下相互配合以保持同步.假如某一浏览器内核的实现至少有三个常驻线程:网页特效引擎线程(用于处理JS),GUI界面渲染线程,浏览器事件触发线程(用于控制交互),除些以外,也有一些执行完就终止的线程,如Http请求线程,这些异步线程都会产生不同的异步事件,下面通过一个图来阐明单线程的JavaScript引擎与另外那些线程是怎样互动通信的.虽然每个浏览器内核实现细节不同,但这其中的调用原理都是大同小异.
由图可看出,浏览器中的JavaScript引擎是基于
事件驱动的,这里的事件可看作是浏览器派给它的各种任务,这些任务可以源自 JavaScript引擎当前执行的代码块,如调用setTimeout添加一个任务,也可来自浏览器内核的其它线程,如界面元素鼠标点击事件,定时
触发器时间到达通知,异步请求状态变更通知等.从代码角度看来任务实体就是各种回调函数,JavaScript引擎一直等待着任务队列中任务的到来.由于单线程关系,这些任务得进行排队,一个接着一个被引擎处理.
上图t1-t2..tn表示不同的时间点,tn下面对应的小方块代表该时间点的任务,假设现在是t1时刻,引擎运行在t1对应的任务方块代码内,在这个时间点内,我们来描述一下浏览器内核其它线程的状态.
t1时刻:
GUI渲染线程:
该线程负责渲染浏览器界面HTML元素,当界面需要重绘(Repaint)或由于某种操作引发回流(reflow)时,该线程就会执行.本文虽然重点解释JavaScript定时机制,但这时有必要说说渲染线程,因为该线程与JavaScript引擎线程是互斥的,这容易理解,因为 JavaScript脚本是可操纵DOM元素,在修改这些元素属性同时渲染界面,那么渲染线程前后获得的元素数据就可能不一致了.
在JavaScript引擎运行脚本期间,浏览器渲染线程都是处于挂起状态的,也就是说被”冻结”了.
所以,在脚本中执行对界面进行更新操作,如添加结点,删除结点或改变结点的外观等更新并不会立即体现出来,这些操作将保存在一个队列中,待JavaScript引擎空闲时才有机会渲染出来.
GUI事件触发线程:
JavaScript脚本的执行不影响html元素事件的触发,在t1时间段内,首先是用户点击了一个鼠标键,点击被浏览器事件触发线程捕捉后形成一个鼠标点击事件,由图可知,对于JavaScript引擎线程来说,这事件是由其它线程异步传到任务队列尾的,由于引擎正在处理t1时的任务,这个鼠标点击事件正在等待处理.
定时触发线程:
注意这里的浏览器模型定时计数器并不是由JavaScript引擎计数的,因为JavaScript引擎是单线程的,如果处于阻塞线程状态就计不了时,它必须依赖外部来计时并触发定时,所以队列中的定时事件也是异步事件.
由图可知,在这t1的时间段内,继鼠标点击事件触发后,先前已设置的setTimeout定时也到达了,此刻对JavaScript引擎来说,定时触发线程产生了一个异步定时事件并放到任务队列中, 该事件被排到点击事件回调之后,等待处理.
同理, 还是在t1时间段内,接下来某个setInterval定时器也被添加了,由于是间隔定时,在t1段内连续被触发了两次,这两个事件被排到队尾等待处理.
可见,假如时间段t1非常长,远大于setInterval的定时间隔,那么定时触发线程就会源源不断的产生异步定时事件并放到任务队列尾而不管它们是否已被处理,但一旦t1和最先的定时事件前面的任务已处理完,这些排列中的定时事件就依次不间断的被执行,这是因为,对于JavaScript引擎来说,在处理队列中的各任务处理方式都是一样的,只是处理的次序不同而已.
t1过后,也就是说当前处理的任务已返回,JavaScript引擎会检查任务队列,发现当前队列非空,就取出t2下面对应的任务执行,其它时间依此类推,由此看来:
如果队列非空,引擎就从队列头取出一个任务,直到该任务处理完,即返回后引擎接着运行下一个任务,在任务没返回前队列中的其它任务是没法被执行的.
相信您现在已经很清楚JavaScript是否可多线程,也了解理解JavaScript定时器运行机制了,下面我们来对一些案例进行分析:
案例1:setTimeout与setInterval
setTimeout(function(){
/* 代码块... */
setTimeout(arguments.callee, 10);
}, 10);
setInterval(function(){
/*代码块... */
}, 10);
这两段代码看一起效果一样,其实非也,第一段中回调函数内的setTimeout是JavaScript引擎执行后再设置新的setTimeout 定时, 假定上一个回调处理完到下一个回调开始处理为一个时间间隔,理论两个setTimeout回调执行时间间隔>=10ms .第二段自setInterval设置定时后,定时触发线程就会源源不断的每隔十秒产生异步定时事件并放到任务队列尾,理论上两个setInterval 回调执行时间间隔<=10.
主要参考:http://www.111cn.net/wy/js-ajax/39717.htm
相关推荐
在JavaScript中,可以使用`setTimeout`和`setInterval`两个函数来模拟多线程的效果。这两个函数可以让一个函数在指定的时间后执行一次,或者每隔指定的时间周期性地执行。 首先,我们来解释一下这两个函数的基本...
虽然原生JavaScript是基于事件循环的单线程语言,但可以通过一些技巧来模拟多线程的效果,例如利用`setTimeout`或`setInterval`等异步机制。本文将介绍一种通过利用GIF图像加载事件来模拟多线程的方法。 #### GIF ...
总的来说,JavaScript通过单线程、事件循环和异步任务队列的机制,实现了看似多线程的效果。理解这一机制对于编写高性能、响应式的Web应用至关重要。开发者需要谨慎处理可能导致阻塞的长时间运行任务,充分利用异步...
本文将详细介绍如何在 C# 中实现类似 JavaScript 的 `setTimeout` 和 `setInterval` 功能,并提供具体的实现代码和注意事项。 #### SetTimeout函数详解 `SetTimeout` 函数的主要作用是在指定的时间间隔后执行一次...
这意味着在任何时候,JavaScript引擎只会执行一个任务,避免了多线程可能导致的竞态条件和死锁问题。 然而,随着Web应用复杂性的增加,单线程模型的限制逐渐显现,如处理大量计算或长时间阻塞的I/O操作时,会阻塞...
然而,JavaScript在浏览器环境中通过异步机制和事件循环来处理多任务,从而给人一种多线程的错觉。JavaScript API提供了诸如`setTimeout`和`setInterval`这样的定时器函数,它们在实际操作中并不立即执行回调函数,...
`timercpp`库的核心原理是基于异步编程模型,它可能采用了多线程、信号量、定时器或者事件驱动(如epoll、kqueue)等技术来实现定时和间隔调用。这个库使得开发者可以更方便地在C++代码中创建延迟执行或周期性执行的...
大概半年前发表过一篇关于setTimeout和setInterval的文章,但是现在回去仔细一看发现其实存在很多不足以及错误。事实上,setTimeout和setInterval并没有我们字面上理解的那么简单。要真正掌握并理解这两个方法,还得...
这种模型避免了多线程可能导致的竞态条件和死锁问题,但同时也带来了挑战,例如阻塞主线程可能导致页面无响应。 ### 二、setTimeout和setInterval 1. **setTimeout**: 这个函数用于在指定的毫秒数后调用一个函数...
1. **非阻塞执行**:JavaScript是单线程的,`setInterval`中的函数执行不会阻塞后续代码。如果函数执行时间过长,可能导致后续的定时器延迟执行。 2. **清除定时器**:使用`clearInterval`可以取消定时器。通常,...
在JavaScript中,`setTimeout`和`setInterval`是两个非常重要的函数,它们分别用于在指定时间后执行一次回调函数和周期性地重复执行回调函数。然而,C#标准库并没有直接提供这两个函数,但我们可以根据JavaScript的...
JavaScript通过事件循环(Event Loop)和异步编程模型来实现多任务处理,尽管它本身不支持真正的多线程。 在JavaScript中,所有代码都是在JavaScript引擎线程中执行的,这个线程负责解释和执行代码。由于JavaScript...
总之,setTimeout和setInterval是JavaScript中非常强大的工具,它们可以帮助开发者在适当的时候执行特定的代码,实现复杂的用户交互和动态功能。然而,在使用这些定时器时,开发者需要注意它们的异步特性以及可能受...
3. **模拟多线程**:由于 JavaScript 的单线程特性,我们无法真正实现多线程,但可以通过一些技巧来模拟多线程的效果,比如利用回调函数、`setTimeout` 或者 `setInterval` 来控制任务的执行顺序和并发度。...
在JavaScript编程中,`setTimeout`和`setInterval`是两个常用的定时器函数,用于在指定的时间后执行一段代码。尽管它们的基本功能相似,但在使用场景和性能上存在差异。`setTimeout`用于执行一次性的延迟操作,而`...
这是为了避免多线程带来的竞态条件和同步问题,从而简化了编程模型。 JavaScript定时器主要包括两种类型:`setTimeout`和`setInterval`。`setTimeout`用于在指定的毫秒数后调用一个函数或执行某段代码,而`...