zoukankan      html  css  js  c++  java
  • JavaScript之事件循环,宏任务与微任务

    首先看一段代码:

    打印顺序是什么?

     正确答案:script start, script end, promise1, promise2, setTimeout

    其中涉及到事件循环(event loop),宏任务(macrotask),微任务(microtask)

    一、事件循环 Event Loop

    程序中设置两个线程:一个负责程序本身的运行,称为"主线程";另一个负责主线程与其他进程(主要是各种I/O操作)的通信,被称为"Event Loop线程"(可以译为"消息线程")。

    所有任务可以分成两种,一种是同步任务(synchronous),另一种是异步任务(asynchronous)。

    同步任务指的是,在主线程上排队执行的任务,只有前一个任务执行完毕,才能执行后一个任务;

    异步任务指的是,不进入主线程、而进入"任务队列"(task queue)的任务,只有"任务队列"通知主线程,某个异步任务可以执行了,该任务才会进入主线程执行。

    一般而言,异步任务有以下三种类型:

      1、普通事件,如click、resize等

     

      2、资源加载,如load、error等

     

      3、定时器,包括setInterval、setTimeout等

    事件循环具体过程就是:

      • 同步任务进入主线程,异步任务进入Event Table并注册函数。
      • 当异步任务完成时,Event Table会将这个函数移入Event Queue。
      • 主线程内的任务执行完毕执行栈为空,会去Event Queue读取对应的函数,进入主线程执行。
      • 上述过程会不断重复,也就是常说的Event Loop(事件循环)。

    二、宏任务与微任务

    在JavaScript中,任务被分为两种,一种宏任务(MacroTask),一种叫微任务(MicroTask)。

    2.1MacroTask(宏任务)

    宿主环境提供的(浏览器和node)

      • script全部代码、setTimeoutsetInterval

    浏览器为了能够使得JS内部task与DOM任务能够有序的执行,会在一个task执行结束后,在下一个 task 执行开始前,对页面进行重新渲染 (task->渲染->task->...)

    2.2MicroTask(微任务)

    语言标准提供的

      • Promise、await

    async函数表示函数里面可能会有异步方法,await后面跟一个表达式,async方法执行时,遇到await会立即执行表达式,然后把await表达式后面的代码放到微任务队列里,让出执行栈让同步代码先执行

    async function foo() {
        var a = await new Promise((resolve) => {
            setTimeout(() => {
                resolve(1);
            }, 2000);
        });
        console.log(a); // 第2秒时输出: 1
    }
    foo();

    2.3宏任务与微任务执行顺序:

      • 执行栈在执行完同步任务后,查看执行栈是否为空,如果执行栈为空,就会去检查微任务队列是否为空,如果为空的话,就执行宏任务,否则就一次性执行完所有微任务。
      • 每次单个宏任务执行完毕后,检查微任务队列是否为空,如果不为空的话,会按照先入先出的规则全部执行完微任务后,设置微任务队列为null,然后再执行宏任务,如此循环。

    总结:同步—>微任务—>宏任务

  • 相关阅读:
    前言(CSDN也有Markdown了,好开森)
    One usage of recurison: the tower of Hanoi
    使用Android注解来改善代码
    mysql生产环境____主从同步修复案例
    不同类型的指针
    C++ 对象模型
    为什么需要模版成员方法
    理解 traits
    C++ 异常处理
    传const引用代替传值
  • 原文地址:https://www.cnblogs.com/vickylinj/p/13596516.html
Copyright © 2011-2022 走看看