javaScript真的是一步的吗?

来源:互联网 发布:最近红的网络歌曲 编辑:程序博客网 时间:2024/04/20 23:41

 先解决我们最初的问题

 for(var i=0; i<3; i++){

 setTimeout(function(){

             alert(i);

         },200)

         }

 alert(i);

答案应该是 3 3 3 3

for循环中变量都是全局的。不像java,在函数内声明的变量,在函数外不能直接访问,js都是全局的, 所以可以访问

所以应该是 3 3 3 3

分析:

 

  JavaScript是单线程执行的无法同时执行多段代码。当某一段代码正在执行的时候,所有后续的任务都必须等待,形成一个队列。一旦当前任务执行完毕,再从队列中取出下一个任务,这也常被称为 “阻塞式执行”。所以一次鼠标点击,或是计时器到达时间点,或是Ajax请求完成触发了回调函数,这些事件处理程序或回调函数都不会立即运行,而是立即排队,一旦线程有空闲就执行。假如当前JavaScript线程正在执行一段很耗时的代码,此时发生了一次鼠标点击,那么事件处理程序就被阻塞,用户也无法立即看到反馈,事件处理程序会被放入任务队列,直到前面的代码结束以后才会开始执行。如果代码中设定了一个setTimeout,那么浏览器便会在合适的时间,将代码插入任务队列,如果这个时间设为0,就代表立即插入队列,但不是立即执行,仍然要等待前面代码执行完毕。所以setTimeout 并不能保证执行的时间,是否及时执行取决于JavaScript 线程是拥挤还是空闲。

如果不理解请通过下面的例子验证

不出意外你们预估的答案应该是先输出 1 再输出 2

但是即使是在setTimeout 延时0秒的情况下 依然是先输出2再输出1

1. function a()  

2. {  

3.     setTimeout(function(){console.log(1);},0);  

4.     console.log(2);  

5. }  

也就是说setTimeout只能保证在指定的时间过后将任务(需要执行的函数)插入队列等候,并不保证这个任务在什么时候执行。执行javascript的线程会在空闲的时候,自行从队列中取出任务然后执行它。javascript通过这种队列机制,给我们制造一个异步执行的假象。

 

 

 

 

 

 

 

原创粉丝点击