zjffun blog

JS 事件循环、任务队列和帧

更新于 写于 前端JavaScript

栈、堆和队列

  • 栈:函数调用形成栈
  • 堆:对象的内容分配在堆中
  • 队列:一个 JavaScript 运行时包含了一个待处理的消息队列。在事件循环期间依次处理队列中的消息

事件循环

之所以称之为事件循环,是因为它经常按照类似如下的方式来被实现:

js
while (queue.waitForMessage()) {
  queue.processNextMessage();
}

如果当前没有任何消息,queue.waitForMessage() 会同步地等待消息到达。

任务队列

任务分为 microtask 和 macrotask。

每次先将 microtask queue 中的任务都处理完,再处理一个 macrotask queue 中的的任务。依次这样循环下去。

例如:

html
<!DOCTYPE html>
<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

frame

event-dispatch

html
<script>
  setTimeout(() => {
    console.log("setTimeout1");
  }, 0);
  requestAnimationFrame(() => {
    console.log("rAF1");
  });
  setTimeout(() => {
    console.log("setTimeout2");
  }, 0);
  requestAnimationFrame(() => {
    console.log("rAF2");
  });
  /**
    setTimeout1
    setTimeout2
    rAF1
    rAF2
   */
</script>

参考

帧: