之前看了一篇关于,Event Loop的文章,做了下面的笔记。
Event Loop 是解决 JavaScript 单线程问题的一种运行机制
当程序接受到了请求之后,就会把请求交给 Event Loop 然后继续往下执行,这样就不用等待请求返回,继续执行下面的代码,当 Event Loop 完成后会把结果返回给程序,程序再调用已经设定好的回调函数,完成任务
但是就会出现了如下的问题
后来我在 stackoverflow 上看到 Understanding the Event Loop 和 tuts+ 里的 Event-Based Programming: What Async Has Over Sync
下面是我对Event Loop的个人理解
“the Event Loop is a queue of callback functions.”
JavaScript执行程序,当执行到异步代码 (async code) 的时候,会把回调函数放在 Event Loop 里,然后继续往下执行,当区块中的其他全部代码执行完之后,再去执行 Event Loop 里面的回调函数
怎么验证呢,我们使用 JavaScript 里最基础的异步函数setTimeout()
console.log('a');
setTimeout(function() {
console.log('b');
}, 0);
console.log('c');
没错 输出为 a c b 这就可以证明上面所说的,回调函数输出b,但一定要等区块里的其他代码执行完后,才可以执行,再看一个例子
console.log('a');
test();
console.log('c');
function test() {
console.log('d');
setTimeout(function() {
console.log('b');
}, 0);
console.log('e');
}
输出 a d e c b 这下就更清楚明白了,所以 Event Loop 其实并不是另一个线程,只是一个放回调函数的队列,等待线程去执行,那就解决了上面的两个问题了,还有一个问题,如果 Event Loop 里有很多回调函数,他们执行的顺序又是怎样呢?
setTimeout(function() {
console.log('a');
}, 20);
setTimeout(function() {
console.log('b');
}, 10);
setTimeout(function() {
console.log('c');
}, 30);
输出是 b a c, web API里的 windowTimers 计算时间,10ms后把console.log('b')
放在了 Event Loop,如此类推。说明 Event Loop 是一个先进先出的队列(FIFO),先放进去的回调函数会先拿出来执行。
整个浏览器可以说就是下面的这张图
What the heck is the event loop anyway? 这个视频解释得很好很明白