栈、堆和队列
- 栈:函数调用形成栈
- 堆:对象的内容分配在堆中
- 队列:一个 JavaScript 运行时包含了一个待处理的消息队列。在事件循环期间依次处理队列中的消息
事件循环
之所以称之为事件循环,是因为它经常按照类似如下的方式来被实现:
1
2
3
while (queue.waitForMessage()) {
queue.processNextMessage();
}
如果当前没有任何消息,
queue.waitForMessage()
会同步地等待消息到达。
任务队列
任务分为 microtask 和 macrotask。
每次先将 microtask queue 中的任务都处理完,再处理一个 macrotask queue 中的的任务。依次这样循环下去。
例如:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
<script>
console.log("main1");
Promise.resolve().then(() => console.log("micro1"));
console.log("main2");
setTimeout(console.log, 0, "macro1");
console.log("main3");
Promise.resolve().then(() => {
console.log("micro2");
Promise.resolve().then(() => console.log("micro3"));
setTimeout(console.log, 0, "macro2");
});
Promise.resolve().then(() => console.log("micro4"));
console.log("main4");
setTimeout(console.log, 0, "macro3");
console.log("main5");
/**
main1
main2
main3
main4
main5
micro1
micro2
micro4
micro3
macro1
macro3
macro2
*/
</script>
- macrotasks: setTimeout, setInterval, setImmediate(Non-standard), I/O, UI rendering
- microtasks: process.nextTick, Promises, Object.observe(Obsolete), MutationObserver
帧
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
<script>
setTimeout(() => {
console.log("setTimeout1");
}, 0);
requestAnimationFrame(() => {
console.log("rAF1");
});
setTimeout(() => {
console.log("setTimeout2");
}, 0);
requestAnimationFrame(() => {
console.log("rAF2");
});
/**
setTimeout1
setTimeout2
rAF1
rAF2
*/
</script>
参考
- Concurrency model and Event Loop - JavaScript | MDN
- javascript - Difference between microtask and macrotask within an event loop context - Stack Overflow
- task-queue - HTML Standard
帧: