zoukankan      html  css  js  c++  java
  • 关于js中事件循环、同步异步、宏任务和微任务存在的时候函数执行顺序的简单理解

    讲述目的:本文章针对JS萌新,是要用最简单的解释让读者能够判断在函数语句的执行顺序,不涉及关于js更深层的理解和探讨也不花费精力讲解js为什么会将执行任务有这些区分,只讨论任务的执行顺序,保护读者不被各种概念绕晕,大神可绕道。

    讲述思路

    1.简单理解同步异步、宏任务和微任务

    2.执行顺序判断方法

    3.简单实例分析

    4.稍复杂点的实例分析

    正文开始

    1.简单理解同步异步、宏任务和微任务

    js是单线程的,所有的任务都要排队挨个执行,就好比做保健(执行js代码),保健师傅只有一个(单线程),顾客(js代码)需排队享受服务,排队的顺序按照顾客的种类(同步异步、宏任务微任务)和顾客到店顺序(在代码中的位置)执行;

    同步与异步、宏任务和微任务分别是函数两个不同维度的描述。

    异步任务:setTimeout和setInterval、ajax、事件绑定等

    同步任务:除了异步任务外的所有任务

    微任务:process.nextTick和 Promise后的theny语句和catch语句等

    宏任务:除了微任务以外的所有任务 

    2.执行顺序判断方法

    先同步再异步,在此基础上先宏任务再微任务

    3.简单实例分析

     1 setTimeout(function () {
     2 new Promise(function (resolve, reject) {
     3 console.log('异步宏任务promise');
     4 resolve();
     5 }).then(function () {
     6 console.log('异步微任务then')
     7 })
     8 console.log('异步宏任务');
     9 }, 0)
    10 new Promise(function (resolve, reject) {
    11 console.log('同步宏任务promise');
    12 resolve();
    13 }).then(function () {
    14 console.log('同步微任务then')
    15 })
    16 console.log('同步宏任务')

    结果

    分析:setTimeout是异步任务,虽然他在0秒后执行但仍排在队列的后面,因此其中的代码全部靠后执行;new Promise是同步任务同时也是主任务,因此第一行先打印'同步宏任务promise',then是微任务所以靠后执行,先执行第16行代码,之后再执行第13行的then语句,因此第二行输出为'同步宏任务',第三行为'同步微任务then';接下来执行setTimeout中的语句,then因为是微任务所以在第8行执行完成后再执行。

     4.稍复杂点的实例分析

     1     setTimeout(() => {
     2         console.log('异步1任务time1');
     3         new Promise(function (resolve, reject) {
     4             console.log('异步1宏任务promise');
     5             setTimeout(() => {
     6                 console.log('异步1任务time2');
     7             }, 0);
     8             resolve();
     9         }).then(function () {
    10             console.log('异步1微任务then')
    11         })
    12     }, 0);
    13     console.log('主线程宏任务');
    14     setTimeout(() => {
    15         console.log('异步2任务time2');
    16 
    17     }, 0);
    18     new Promise(function (resolve, reject) {
    19         console.log('宏任务promise');
    20         // reject();
    21         resolve();
    22     }).then(function () {
    23         console.log('微任务then')
    24     }).catch(function () {
    25         console.log('微任务catch')
    26     })
    27     console.log('主线程宏任务2');

    本例中需注意第9行的then是在第14行的setTimeout之前执行的,而第5行的setTimeout在第14行之后执行。也就是在一个异步任务代码块中,会先执行完所有同步语句(包括宏任务和微任务),然后去执行整个代码中的同级别的异步任务,而第5行的setTimeout因是第二层异步语句,会被放到之后才执行。

    执行结果为

  • 相关阅读:
    WCF 、Web API 、 WCF REST 和 Web Service 的区别
    BusyIndicator using MVVM 忙碌状态指示器的的实现
    复制文件夹的方法 .net
    SQL/LINQ/Lamda
    CSLA验证规则总结
    C++中GB2312字符串和UTF-8之间的转换
    如何用VC编写供PB调用的DLL
    【转】lucene4.3.0 配置与调试
    cygwin主要命令
    【转】eclipse中window->preference选项中没有tomcat的解决方法
  • 原文地址:https://www.cnblogs.com/jinpanpan/p/10883548.html
Copyright © 2011-2022 走看看