都2022年了你不会还没搞懂js异步编程吧

本文涉及的产品
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: js异步编程

简介

在说js异步编程之前,我们先来说说同步和异步的概念。

同步,可以理解为在执行完一个函数或方法之后,一直等待系统返回值或消息,这时程序是处于阻塞的,只有接收到返回的值或消息后才往下执行其他的命令。  

异步,执行完函数或方法后,不必阻塞性地等待返回值或消息,只需要向系统委托一个异步过程,那么当系统接收到返回值或消息时,系统会自动触发委托的异步过程,从而完成一个完整的流程。 

js中对于异步任务都有哪些解决方案呢?笔者今天就来说说js中的常见的五种异步解决方案。

image.png

异步解决方案

js中常用的异步解决方案有以下五种

  1. 回调函数(callback)
  2. 事件监听
  3. Promise
  4. Generator
  5. Async/Await

回调函数(callback)

回调函数简单理解就是一个函数被作为参数传递给另一个函数。回调函数是异步最简单的一种解决方案。

比如下面这个例子,我们想在请求后异步输出我们从后端获取的结果,就可以使用到回调函数。

function fn1(callback){
  // 模拟ajax请求获取后端数据,耗时一秒
  setTimeout(() => {
    const data = '我是后端返回的结果'
    callback && callback(data)
  }, 1000)
}

fn1((data)=>{
  console.log(data)
})

在上面的例子中(data)=>{console.log(data)}就是回调函数,该函数会被当做参数传递给另外一个函数,当异步任务有了结果就会执行该回调函数,并不会阻碍主程序的执行,所以回调函数是js中最开始使用的异步解决方案。

使用回调函数的缺点也很明显,把函数当做参数传递,不利于代码的阅读和维护,各个部分之间高度耦合。而且如果有多个回调函数嵌套使用的话容易造成回调地狱

比如

fun1(() => {
  fun2(() => {
    fun3()
  })
})

事件监听

事件监听在笔者前面的文章js中的事件中有介绍,不了解的同学可以再看看。我们可以利用自定义事件并监听来实现异步。

前面我们已经介绍了创建自定义事件有两种方法,所以在不同情况下我们可以使用不同的方法。

使用new Event()创建自定义事件,这种方式不能传递参数,适用于简单的场景。

const myEvent = new Event("test");

// 监听事件
document.addEventListener("test", function (e) {
  // 类似回调函数,可以在这里进行相应的处理
  console.log("自定义事件触发了");
});

// 触发自定义事件
setTimeout(function () {
  if (document.dispatchEvent) {
    document.dispatchEvent(myEvent);
  } else {
    // 兼容低版本浏览器
    document.fireEvent(myEvent);
  }
}, 2000);

使用new CustomEvent()创建自定义事件,这种方式可以传递参数,适用于需要传递参数的场景。

// 创建自定义事件 能传递参数,必须使用detail作为key不然获取不到
const myEvent2 = new CustomEvent("test2", { detail: { name: "randy" } });

// 监听事件
document.addEventListener("test2", function (e) {
  // 类似回调函数,可以在这里进行相应的处理
  console.log("自定义事件触发了参数是", e.detail);
});

// 触发自定义事件
setTimeout(function () {
  if (document.dispatchEvent) {
    document.dispatchEvent(myEvent2);
  } else {
    // 兼容低版本浏览器
    document.fireEvent(myEvent2);
  }
}, 3000);

使用事件监听是异步的一个解决方案,但是整个程序变成了事件驱动,并且每次使用还得注册事件再监听再进行触发使得我们使用的时候比较麻烦(代码量多),并且阅读代码的时候,各部分比较分散,很难看出主流程。

Promise

在ES2015 (ES6)中引入了Promise对象,它是异步编程中一种比较好的解决方案。

Promise特点

Promise对象代表一个异步操作,它有三种状态

  • 进行中 (Pending)
  • 已完成 (Resolved/Fulfilled)
  • 已失败 (Rejected)

只有异步操作的结果,可以决定当前是哪一种状态,任何其他操作都无法改变这个状态。

Promise对象状态一旦改变,就不会再变,Promise对象的状态改变,只有两种可能

  • 从Pending变为Resolved
  • 从Pending变为Rejected

只要这两种情况发生,状态就凝固,不会再变了,会一直保持这个结果。

Promise相关API

new Promise()

Promise是一个构造函数,我们可以通过new关键字来创建一个Promise实例,也可以直接使用Promise的一些静态方法。

new Promise((resolve, reject) => {...});

处理器函数接收两个参数分别是resolvereject,这两个参数也是两个回调函数

resolve 函数在异步操作成功时调用,并将异步操作的结果,作为参数传递出去

reject 函数在异步操作失败时调用,并将异步操作报出的错误,作为参数传递出去

简单理解就是一个是成功回调,一个是失败回调。

比如下面的例子,当随机数小于5我们就返回成功,否则就返回失败。

function fun1() {
  return new Promise((resolve, reject) => {
    const num = Math.ceil(Math.random()*10)
    if(num < 5){
      resolve(num)
    }else{
      reject('数字太大')
    }
  })
}

我们还可以使用Promise的一些静态方法来创建Promise对象,上面的例子我们可以改写为

function fun2() {
  const num = Math.ceil(Math.random()*10)
  if(num < 5){
    return Promise.resolve(num)
  }else{
    return Promise.reject('数字太大')
  }
}

Promise.prototype.then()

Promise实例生成以后,可以用then方法指定resolved状态和reject状态的回调函数。

Promise.prototype.then(onFulfilled[, onRejected])

第一个参数是resolved状态的回调函数,第二个参数是reject状态的回调函数(一般我们不使用而是使用catch方法)。

fun1().then(
  (res) => {
    // res是fun1方法resolve的参数,即小于5的随机数
    console.log(res);
  },
  (err) => {
    // res是fun1方法reject的参数,即数字太大
    console.log(err);
  }
);

前面说到我们一般不使用而是使用then方法的第二个参数,而是使用catch方法,所以上面的例子我们还可以改写为

fun1()
  .then((res) => {
    console.log(res);
  })
  .catch((err) => {
    console.log(err);
  });

说到这,很多小伙伴会好奇,为什么不使用then方法的第二个参数而是使用catch方法来处理错误呢?我们知道Promise是链式调用的,实际使用过程中可能会有很多的then方法一直调用下去,所以如果我们把错误处理都写在then方法中的话就需要在每个then中对错误进行处理了。而使用catch方法的话我们只需要在链式调用的最末尾加上一个catch方法就可以啦,就可以捕捉到第一个错误啦。这样会大大简化我们的代码。

then 方法必须返回一个 promise 对象

  1. 如果then方法中返回的是一个普通值(如Number、String等)就使用此值包装成一个新的Promise对象返回。
function fun3() {
  return Promise.resolve(1);
}

fun3()
  .then((res) => {
    console.log(res); // 1
    return "success"; // 返回的是一个普通值(如Number、String等)
  })
  .then((res) => {
    console.log(res); // success
  });
  1. 如果then方法中返回了一个Promise对象,那就以这个对象为准,返回它的结果。
function fun4() {
  return Promise.resolve(1);
}

fun4()
  .then((res) => {
    console.log(res); // 1
    return Promise.resolve("success2");
  })
  .then((res) => {
    console.log(res); // success2
  });
  1. 如果then方法中没有return语句,就返回一个用undefined包装的Promise对象。
function fun5() {
  return Promise.resolve(1);
}

fun5()
  .then((res) => {
    console.log(res); // 1
  })
  .then((res) => {
    console.log(res); // undefined
  });
  1. 如果then方法中出现异常,则会流转到下一个thenonRejected或者最末尾的catch方法中。
function fun6() {
  return Promise.resolve("fun6");
}

fun6()
  .then(
    (res) => {
      console.log("then1 resolve方法" + res);
      return Promise.reject("error1");
    },
    (err) => {
      console.log("then1 reject方法" + err);
    }
  )
  .then(
    (res) => {
      console.log("then2 resolve方法" + res);
    },
    (err) => {
      console.log("then2 reject方法" + err);
    }
  );

上面的例子会输出 then1 resolve方法 fun6、then2 reject方法 error1

function fun6() {
  return Promise.resolve("fun6");
}

fun6()
  .then((res) => {
    console.log("then1 resolve方法" + res);
    return Promise.reject("error1");
  })
  .then((res) => {
    console.log("then2 resolve方法" + res);
    return Promise.reject("error2");
  })
  .catch((err) => {
    console.log("catch方法" + err);
  });

上面的例子会输出 then1 resolve方法 fun6、catch方法 error1

  1. 如果then方法没有传入任何回调,则继续向下传递(即所谓的值穿透)
function fun7() {
  return Promise.resolve("fun7");
}

fun7()
  .then()
  .then()
  .then((res) => {
    console.log(res); // fun7
  });

Promise.prototype.catch()

catch方法前面也有介绍到,主要是用来捕获错误的。此方法也会返回一个新Promise对象。

如果catch方法里再有错误则通过 catch 返回的Promise是一个rejected实例,否则它就是一个成功的resolved实例。

function fun8() {
  return Promise.reject("fun8");
}

fun8()
  .catch((err) => {
    console.log(err); // fun8
    // 如果不抛错则会进入下一个then方法的resolve回调方法,否则会进入reject回调方法
    // throw new Error("11"); 
  })
  .then(
    (res) => {
      console.log("resolve" + res);
    },
    (err) => {
      console.log("reject" + err);
    }
  );

相较于使用then的第二个参数,笔者还是推荐使用catch方法来获取错误。

Promise.prototype.finally()

finally,英文是最后的意思,此方法是ES2018新增的。

finally,在Promise结束时,不管成功还是失败都将执行并且该回调方法无参数。

function fun9() {
  return Promise.reject("fun9");
}

fun9()
  .then((res) => {
    console.log(res);
  })
  .catch((err) => {
    console.log(err);
  })
  .finally(() => {
    console.log("不管resolve还是reject我都会执行");
  });

该方法适用于同时需要在then()catch()中各写一次的情况,比如我们不管请求成功与否都需要关闭按钮loading状态,我们就可以在finally回调方法中设置loading=false来关闭loading状态。

Promise.resolve()

这个方法是一个静态方法,前面在介绍new Promise()构造函数的时候我们也简单介绍了下,该方法接收一个参数并转换为Promise对象。

Promise.resolve(value)

// 类似于
new Promise((resolve, reject) => {
  resolve(value)
})

Promise.reject()

这个方法是一个静态方法,前面在介绍new Promise()构造函数的时候我们也简单介绍了下,该方法接收一个参数并转换为Promise对象。

Promise.reject(value)

// 类似于
new Promise((resolve, reject) => {
  reject(value)
})

Promise.all()

Promise.all(iterable) 参数为一组 Promise 实例组成的数组,用于将多个Promise实例包装成一个新的 Promise实例。

当数组中的Promise实例都为都 Resolved 的时候,Promise.all() 的状态才会 Resolved,否则为Rejected。并且Rejected是第一个被RejectedPromise 的返回值。

function fun10() {
  return Promise.resolve("fun10");
}
function fun11() {
  return Promise.resolve("fun11");
}
function fun12() {
  return Promise.resolve("fun12");
}
Promise.all([fun10(), fun11(), fun12()])
  .then((res) => {
    console.log("all resolve", res); // ['fun10', 'fun11', 'fun12']
  })
  .catch((err) => {
    console.log("all reject", err);
  });

image.png

function fun10() {
  return Promise.resolve("fun10");
}
function fun11() {
  return Promise.resolve("fun11");
}
function fun12() {
  return Promise.reject("fun12");
}
Promise.all([fun10(), fun11(), fun12()])
  .then((res) => {
    console.log("all resolve", res);
  })
  .catch((err) => {
    console.log("all reject", err); // all reject fun12
  });

image.png

Promise.race()

race方法与all方法不同,race方法中只要对象中有一个状态改变了,它的状态就跟着改变,并将那个改变状态实例的返回值传递给回调函数。

也就是不管失败与成功第一个Promise的返回值就是race方法的返回值。resolve就会进入then方法,否则进入catch方法

function fun13() {
  return new Promise(function (resolve, reject) {
    setTimeout(() => {
      return resolve("fun13");
    }, 2000);
  });
}
function fun14() {
  return new Promise(function (resolve, reject) {
    setTimeout(() => {
      return reject("fun14");
    }, 1000);
  });
}

Promise.race([fun13(), fun14()])
  .then((res) => {
    console.log("race resolve " + res); 
  })
  .catch((err) => {
    console.log("race reject " + err); // race reject fun14
  });

image.png

Promise.allSetted()

我们都知道 Promise.all() 具有并发执行异步任务的能力。但它的最大问题就是如果其中某个任务出现异常reject,所有任务都会挂掉,Promise直接进入 reject 状态。

Promise.allSettled()方法就是不管是否成功失败,都会返回结果并进入then方法。

function fun13() {
  return new Promise(function (resolve, reject) {
    setTimeout(() => {
      return resolve("fun13");
    }, 2000);
  });
}
function fun14() {
  return new Promise(function (resolve, reject) {
    setTimeout(() => {
      return reject("fun14");
    }, 1000);
  });
}

Promise.allSetted([fun13(), fun14()])
  .then((res) => {
    console.log("allSetted resolve" + res); //[{...},{...}]
  })
  .catch((err) => {
    console.log("allSetted reject" + err);
  });

image.png

Promise.any()

Promise.any()Promise.race()类似都是返回第一个结果,但是Promise.any()只返回第一个成功的,尽管某个 promisereject 早于另一个 promiseresolve,仍将返回那个最先 resolve 的 promise。

如果都被reject则会抛出All promises were rejected错误。

function fun13() {
  return new Promise(function (resolve, reject) {
    setTimeout(() => {
      return resolve("fun13");
    }, 2000);
  });
}
function fun14() {
  return new Promise(function (resolve, reject) {
    setTimeout(() => {
      return reject("fun14");
    }, 1000);
  });
}

Promise.any([fun13(), fun14()])
  .then((res) => {
    console.log("any resolve" + res); //any resolve fun13
  })
  .catch((err) => {
    console.log("any reject" + err);
  });

image.png

Promise用同步的方式写异步的代码,避免了层层嵌套的回调函数,并且提供了很多api使我们用起来也更加方便。但是Promise的链式调用一直拼接代码也不太优雅。

Generator

Generator也是在在ES2015 (ES6)中引入的,它也是异步编程的一种解决方案,最大的特点就是可以交出函数的执行权。

Generator函数需要和yield关键字配合使用。

语法

*用来表示函数为 Generator 函数,并且只能使用在function函数,不能用在箭头函数中。

函数内部有yield字段,yield用来定义函数内部的状态,并让出执行权,这个关键字只能出现在生成器函数体内,但是生成器中也可以没有 yield 关键字,函数遇到 yield 的时候会暂停,并把 yield 后面的表达式结果抛出去。

调用 Generator 函数和调用普通函数一样,在函数名后面加上()即可,但是Generator 函数不会像普通函数一样立即执行,而是返回一个指向内部状态对象的指针,类似迭代器对象 Iterator 的 next 方法,我们需要手动调用next方法来进行下一步操作,指针就会从函数头部或者上一次停下来的地方开始执行。

这里可能会涉及到迭代器(Iterator),不了解的可以先看看笔者前面写的对象数组遍历 Iterator章节

// 写法很多,function* fn()、function*fn()和function *fn()都可以
function* generatorFn() {
  console.log("a");
  yield "1";
  console.log("b");
  yield "2";
  console.log("c");
  return "3";
}

const generatorIt = generatorFn();
console.log(generatorIt.next()); // a {value: '1', done: false}
console.log(generatorIt.next()); // b {value: '2', done: false}
console.log(generatorIt.next()); // c {value: '3', done: true}

分析Generator函数的执行过程

上面的代码会依次输出a、{value: '1', done: false}、 b、 {value: '2', done: false}、 c、 {value: '3', done: true}下面我们来分析下Generator 函数的执行过程

  1. 首先Generator 函数执行,返回了一个指向内部状态对象的指针generatorIt,此时没有任何输出。
  2. 第一次调用next方法,从 Generator 函数的头部开始执行,先是打印了 a ,执行到yield就停下来,并执行紧跟着yield后边的表达式,将表达式的值 '1'作为返回对象的 value 属性值,此时函数还没有执行完,返回对象的 done 属性值是 false
  3. 第二次调用next方法,从第一个yield 下面开始运行,先是打印了 b ,执行到第二个yield就停下来,并执行紧跟着yield后边的表达式,将表达式的值 '2'作为返回对象的 value 属性值,此时函数还没有执行完,返回对象的 done 属性值是 false
  4. 第三次调用next方法,从第二个yield 下面开始运行,先是打印了 c ,然后执行了函数的返回操作,并将 return 后面的表达式的值,作为返回对象的 value 属性值,此时函数已经结束,所以 done 属性值为true。在这一步如果函数没有返回值这里就会返回{value: undefined, done: true}

说到这里关于Generator函数是不是有了初步的理解呢?简单的理解就是Generator函数yield放到哪里它就停到哪里(注意 紧跟在yield后面的语句是会执行的,并且其返回值是作为迭代对象的value),调用时使用next方法一步一步往后走。

再次分析Generator函数的执行过程

关于yield后面的语句是在当次执行还是next后执行呢?很多小伙伴分不清楚,所以笔者再举个例子来分析下。

function* generatorFn2() {
  console.log("a");
  yield console.log("a2");
  console.log("b");
  yield console.log("b2");
  console.log("c");
}

const generatorIt2 = generatorFn2();
console.log(generatorIt2.next()); // {value: undefined, done: false}
console.log(generatorIt2.next()); // {value: undefined, done: false}
console.log(generatorIt2.next()); // {value: undefined, done: true}

上面的代码会依次输出a、a2、{value: undefined, done: false}、 b、b2、 {value: undefined, done: false}、 c、 {value: undefined, done: true},所以通过这里例子我们可以看出,紧跟在yield后面的语句是在当次执行的。

for..of 遍历Generator

对象数组遍历 Iterator章节里面笔者已经说过了,只要实现了迭代器方法就能使用for of遍历,所以我们可以使用for of来运行我们的Generator函数,所以不再需要我们手动调用next方法啦,一旦next方法的返回对象的done属性为truefor...of循环就会中止,且不包含该返回对象。怎么理解这里的且不包含该返回对象呢? 就是当donetrue时就会终止循环所以最后一项是不会输出出来的。

function* generatorFn() {
  console.log("a");
  yield "1";
  console.log("b");
  yield "2";
  console.log("c");
  return "3";
}

for (const iterator of generatorFn()) {
  console.log(iterator); // a、1、b、2、c
}

上面的例子会依次输出a、1、b、2、c,并没有输出3,所以就印证了当donetrue时就会终止循环所以最后一项是不会输出出来的。

所以Generator函数可以通过for of自动执行,不再需要手动调用next方法,但是最后一项是不会输出来的,这里需要注意下。

Generator函数传参

有小伙伴说既然yield有返回值,是不是我们在代码中通过const res = yield "1";输出res就能得到{value: '1', done: false}呢?下面我们再来分析下

function* generatorFn3() {
  console.log("a");
  const res = yield "1";
  console.log(res); // undefined
}

const generatorIt3 = generatorFn3();
generatorIt3.next()
generatorIt3.next()

上面的代码会依次输出a、undefined,并没有跟我们想象的一样输出a、{value: '1', done: false}而是输出了a、undefined。那我们需要怎样传参呢?这就需要用到next方法啦。我们把上面的例子再改下。

function* generatorFn3() {
  console.log("a");
  const res = yield "1";
  console.log(res); // randy
}

const generatorIt3 = generatorFn3();
generatorIt3.next()
generatorIt3.next('randy') // 通过next方法传参

上面的代码会依次输出a、randy。所以参数可以通过下一次的next来传递,也就是说当 next 传入参数的时候,该参数会作为上一步yield的返回值。

yield* 表达式

yield命令后面加上星号,表明它返回的是一个遍历器,这被称为yield*表达式。

function *foo(){
  yield "foo1"
  yield "foo2"
}
function *bar(){
  yield "bar1"
  yield* foo()
  yield "bar2"
}
for(let val of bar()){
  console.log(val)
}

// bar1 foo1 foo2 bar2

yield命令后面如果不加星号,返回的是整个数组,加了星号就表示返回的是数组的遍历器

function* gen1(){
  yield ["a", "b", "c"]
}
for(let val of gen1()){
  console.log(a)
}
// ["a", "b", "c"]

function* gen2(){
  yield* ["a", "b", "c"]
}
for(let val of gen2()){
  console.log(a)
}
// a b c

Generator中的return

return 方法返回给定值,并结束遍历 Generator 函数,当 return 无没有传值时,就返回 undefined

function* foo() {
  yield 1;
  yield 2;
  yield 3;
}

var f = foo();
console.log(f.next());
// 输出{value: 1, done: false}

console.log(f.return("hahaha"));
// 由于调用了return方法,所以遍历已结束,done变为true 输出{value: "hahaha", done: true}

console.log(f.next());
// 再次调用也只能输出{value: undefined, done: true}

Generator函数优雅的流程控制方式,可以让函数在指定位置可中断执行,但是实际开发过程中我们很少单独Generator函数,因为手动一次次运行next方法十分麻烦,一般我们都会搭配执行器(如 co 库)来运行Generator函数,所以对于单纯解决异步问题还是不太好用。

Generator中的throw

可以通过 throw 方法在 Generator 外部控制内部执行的“终断”。也就是我们常说的抛出异常。

function* gen() {
    while (true) {
        try {
            yield 42
        } catch (e) {
            console.log(e.message)
        }
    }
}

let g = gen()
console.log(g.next()) // { value: 42, done: false }
console.log(g.next()) // { value: 42, done: false }
console.log(g.next()) // { value: 42, done: false }
// 中断操作
g.throw(new Error('break'))

console.log(g.next()) // {value: undefined, done: true}

Async/Await

ES2017(ES8)中引入了 async 函数,使得异步操作变得更加方便。Async/Await 的出现,被很多人认为是js异步操作的最终且最优雅的解决方案。我们可以简单理解Async/Await = Generator + Promise

语法

async 用于声明一个 function 是异步的,await 用于等待一个异步方法执行完成,只有当异步完成后才会继续往后执行。await不是必须的并且await 只能出现在 async 函数中。

async function() {
  const result = await getData()
  console.log(result)
}

一个函数如果加上 async ,那么该函数就会返回一个 Promise

async function async1() {
  return "1"
}
console.log(async1()) // -> Promise {<resolved>: "1"}

这种用书写同步代码的方式处理异步是不是很舒服呢。

错误处理

Async/Await没有Promise那么多的api,错误需要自己使用try catch处理。

async function() {
  try{
    const result = await getData()
    console.log(result)
  } catch(e) {
    console.log(e)
  }
}

Async/Await和Promise对比

  1. Async/Await相较于Promise的链式调用完全用书写同步代码的方式处理异步使代码分厂优雅易懂。
  2. Async/Await这种用书写同步代码的方式使得await 会阻塞后面代码正常运行,也许之后的异步代码并不依赖于前者,但仍然需要等待前者完成,导致代码失去了并发性。

下面笔者使用Async/AwaitPromise作为对比举个例子说明。

function getData() {
  return Promise.resolve("模拟获取后端数据");
}

async function fun1() {
  console.log("主程序开始执行");
  const result = await getData();
  console.log(result);
  console.log("让异步代码自己去执行,不阻塞我们主程序");
}

fun1(); // 主程序开始执行、模拟获取后端数据、让异步代码自己去执行,不阻塞我们主程序

async function fun2() {
  console.log("主程序开始执行");
  getData().then((result) => {
    console.log(result);
  });
  console.log("让异步代码自己去执行,不阻塞我们主程序");
}

fun2(); // 主程序开始执行、让异步代码自己去执行,不阻塞我们主程序、模拟获取后端数据

从上面的例子我们可以看出使用Async/Await的弊端,就是不管后面依不依赖异步结果,Async/Await都一定会阻塞后面代码的执行。

Async/Await和Generator对比

  1. Async/Await内置执行器。 Generator 函数的执行必须靠执行器(如co 函数库),而 Async/Await 函数自带执行器。也就是说,Async/Await 函数的执行,与普通函数一模一样
  2. Async/Await更好的语义。 asyncawait,比起星号和 yield,语义更清楚了。async 表示函数里有异步操作,await 表示紧跟在后面的表达式需要等待结果。

总结

本文笔者一共介绍了五种js异步解决方案,每种方案都有各自的特点,在不同情况下我们可以选择最合适的方案来使用。总体来说笔者觉得Async/AwaitPromise这两种方案目前是最好的异步解决方案。

扩展

async defer

async defer关键字都是用来修饰<script>标签的,但是这三者都有什么区别呢?下面笔者来分析下。

<script src='xxx'></script>
<script src='xxx' async></script>
<script src='xxx' defer></script>

浏览器在解析 HTML 的时候,如果遇到一个没有任何属性的 script 标签,就会暂停解析,先发送网络请求获取该 js 脚本的代码内容,然后让 js 引擎执行该代码,当代码执行完毕后恢复解析。整个过程如下图所示:

当浏览器遇到带有 async 属性的 script 时,请求该脚本的网络请求是异步的,不会阻塞浏览器解析 HTML,一旦网络请求回来之后,如果此时 HTML 还没有解析完,浏览器会暂停解析,先让 js 引擎执行代码,执行完毕后再进行解析,图示如下:

当浏览器遇到带有 defer 属性的 script 时,获取该脚本的网络请求也是异步的,不会阻塞浏览器解析 HTML,一旦网络请求回来之后,如果此时 HTML 还没有解析完,浏览器不会暂停解析并执行 js 代码,而是等待 HTML 解析完毕(DOMContentLoaded事件触发之前)再执行 js 代码,图示如下:

根据上面的分析,不同类型 script 的执行顺序及其是否阻塞解析 HTML 总结如下:

script 标签 JS 执行顺序 是否阻塞解析 HTML
<script> 在 HTML 中的顺序 阻塞
<script async> 网络请求返回顺序 可能阻塞,也可能不阻塞
<script defer> 在 HTML 中的顺序 不阻塞

系列文章

都2022年了你不会还没搞懂JS数据类型吧

都2022年了你不会还没搞懂JS原型和继承吧

都2022年了你不会还没搞懂JS赋值拷贝、浅拷贝、深拷贝吧

都2022年了你不会还没搞懂对象数组的遍历吧

都2022年了你不会还没搞懂this吧

都2022年了你不会还没搞懂JS Object API吧

都2022年了你不会还没搞懂js垃圾回收和内存泄露吧

都2022年你不会还没搞懂js执行上下文和事件循环机制吧

都2022年了你不会还没搞懂js中的事件吧

都2020年了你不会还没搞懂js异步编程吧

后记

感谢小伙伴们的耐心观看,本文为笔者个人学习笔记,如有谬误,还请告知,万分感谢!如果本文对你有所帮助,还请点个关注点个赞~,您的支持是笔者不断更新的动力!

相关文章
|
2月前
|
缓存 JavaScript 前端开发
掌握现代JavaScript异步编程:Promises、Async/Await与性能优化
本文深入探讨了现代JavaScript异步编程的核心概念,包括Promises和Async/Await的使用方法、最佳实践及其在性能优化中的应用,通过实例讲解了如何高效地进行异步操作,提高代码质量和应用性能。
|
2月前
|
JavaScript 前端开发 开发者
探索Node.js中的异步编程之美
在数字世界的海洋中,Node.js如同一艘灵活的帆船,以其独特的异步编程模式引领着后端开发的方向。本文将带你领略异步编程的魅力,通过深入浅出的讲解和生动的代码示例,让你轻松驾驭Node.js的异步世界。
|
2月前
|
JavaScript API 开发者
深入理解Node.js中的事件循环和异步编程
【10月更文挑战第41天】本文将通过浅显易懂的语言,带领读者探索Node.js背后的核心机制之一——事件循环。我们将从一个简单的故事开始,逐步揭示事件循环的奥秘,并通过实际代码示例展示如何在Node.js中利用这一特性进行高效的异步编程。无论你是初学者还是有经验的开发者,这篇文章都能让你对Node.js有更深刻的认识。
|
2月前
|
前端开发 JavaScript UED
探索JavaScript的异步编程模式
【10月更文挑战第40天】在JavaScript的世界里,异步编程是一道不可或缺的风景线。它允许我们在等待慢速操作(如网络请求)完成时继续执行其他任务,极大地提高了程序的性能和用户体验。本文将深入浅出地探讨Promise、async/await等异步编程技术,通过生动的比喻和实际代码示例,带你领略JavaScript异步编程的魅力所在。
32 1
|
2月前
|
前端开发 JavaScript 开发者
除了 async/await 关键字,还有哪些方式可以在 JavaScript 中实现异步编程?
【10月更文挑战第30天】这些异步编程方式在不同的场景和需求下各有优劣,开发者可以根据具体的项目情况选择合适的方式来实现异步编程,以达到高效、可读和易于维护的代码效果。
|
2月前
|
前端开发 JavaScript 开发者
深入理解JavaScript异步编程
【10月更文挑战第29天】 本文将探讨JavaScript中的异步编程,包括回调函数、Promise和async/await的使用。通过实例代码和解释,帮助读者更好地理解和应用这些技术。
32 3
|
2月前
|
前端开发 JavaScript 开发者
除了 Generator 函数,还有哪些 JavaScript 异步编程解决方案?
【10月更文挑战第30天】开发者可以根据具体的项目情况选择合适的方式来处理异步操作,以实现高效、可读和易于维护的代码。
|
2月前
|
前端开发 JavaScript
深入理解 JavaScript 的异步编程
深入理解 JavaScript 的异步编程
39 0
|
3月前
|
前端开发 JavaScript UED
探索JavaScript中的异步编程模式
【10月更文挑战第21天】在数字时代的浪潮中,JavaScript作为一门动态的、解释型的编程语言,以其卓越的灵活性和强大的功能在Web开发领域扮演着举足轻重的角色。本篇文章旨在深入探讨JavaScript中的异步编程模式,揭示其背后的原理和实践方法。通过分析回调函数、Promise对象以及async/await语法糖等关键技术点,我们将一同揭开JavaScript异步编程的神秘面纱,领略其带来的非阻塞I/O操作的魅力。让我们跟随代码的步伐,开启一场关于时间、性能与用户体验的奇妙之旅。
|
2月前
|
JavaScript 前端开发
深入理解Node.js中的异步编程模型
【10月更文挑战第39天】在Node.js的世界里,异步编程是核心所在,它如同自然界的水流,悄无声息却又无处不在。本文将带你探索Node.js中异步编程的概念、实践以及如何优雅地处理它,让你的代码像大自然的流水一样顺畅和高效。