学习关键语句 :
async , await 用法
await 怎么使用
同步任务和异步任务
微任务和宏任务
js中代码执行顺序
虽然说 async 和 await 是 Promise 的语法糖 , 但是用惯了Promise 的人(我) , 还真不能超快速使用上这个语法糖 , 所以赶紧写一篇文章出来让各位了解了解这个到底怎么用在我的项目中呢 , 毕竟大家都是想用在项目里的 , 而不是说简单的学习一下语法 难道只有我看了语法介绍还是不会用嘛
同时 这一部分和异步任务有很大的关系 , 因为一般 await 我们是在发起请求时使用的 , 所有我们同样也会看看 js 中的同步任务和异步任务
而且我们从异步任务开始会更好的理解了 await 的用法
而异步任务需要从同步任务开始
这篇文章的目的就是希望用最简单的话来说明白上面这一些东东
警告 ! 本文可能带有知识误导 , 得出的结论相不相信取决于自己 , 作者说.
低情商 : 不会吧 , 不会真有人来看同步任务是啥吧 ?
低情商解答 : 上一行代码不执行完下一行代码就不开始执行 , 这就是同步 , 而执行就是任务
高情商 : 兄弟们又来复习啦 , 温故而知新啊
高情商解答 : JavaScript 是一门单线程语言 , 这意味着对于你写的 js 代码 , 他的执行顺序必然是从上往下一行一行执行的 , 一旦哪行出错了 , 那整个程序都会停止运行 , 所以一行一行做任务我们称之为同步任务
异步任务简单来说 , 就是他不是同步任务
啊这是在说 , 结合上面同步任务的意思 , 异步任务就不会一行一行的执行 , 他超脱了原本的代码执行顺序 , 异步任务有自己执行的时候 , 这个一会儿再看什么时候执行
异步任务分为两种 , 微任务 和 宏任务 , 列举一些常见的
微任务
Promise.then async await
宏任务
setTimeout setInterval
对于这一块只需记住 , 从执行顺序上微任务是要先于宏任务的
我们现在已经知道了 Promise.then 是一种微任务 , 同时我们还知道微任务是异步任务的一种 , 并且你还突然想起来异步任务不是同步任务 淦整天记的都啥
我们来看一段简单的代码 , 以下代码可以直接在浏览器按 F12 在控制台开始测试
function getP() {
return new Promise((resolve, reject) => {
console.log(0)
resolve()
})
}
getP().then(() => console.log(1))
console.log(2)
解释:
已知 : 按照顺序先执行 Promise() , 再执行 .then() , 最后执行 console.log(2)
但是又已知 .then() 是微任务 , 所以就跳出了同步任务 , 在最后执行打印 1 的任务
众所周知 , 我们在发起请求之后 , 就会在回调中使用返回的数据 , 这就是最常见的使用场景了
我们看看下面这段代码
function getP() {
return new Promise((resolve, reject) => {
resolve({data:{id:1}})
})
}
let userId = -1
getP().then(res => userId = res.data.id)
我们为了获得用户的 Id , 必须向后台发起请求 , 并在回调中赋值给页面中的数据
而有时候我们需要先通过上一个请求获取到数据 , 再使用这个数据进行下一次请求 , 简单来说就是请求的嵌套
请看下面这段代码 , 这是正常使用 promise.then 来获取数据再通过数据发起请求的做法
function getP(id = -1) {
console.log(id)
return new Promise((resolve, reject) => {
resolve({data:{id:1}})
})
}
let userId = -1
getP().then(res => {
userId = res.data.id
getP(userId)
.then(() => {})
})
其实这样倒也没什么问题 , 只是如果需要请求的次数变多 , 还是会出现类似于回调地狱的情况 , 所以才会又出现了 async 和 await 的语法糖
关于这俩只需要记住关键的点就足够了
- async 和 await 是成双成对的 , 但是使用 async 可以不带 await , 使用 await 必须带上 async
- 使用 async 修饰的函数必定返回一个 promise 对象 , 如果该函数没有返回值 , 那返回的 promise 对象的成功回调参数为 undefined ; 如果有返回值且不为 promise , 那么返回值是什么成功回调参数就是什么 ; 如果返回值是 promise , 那就不用再说了
- 使用 await 修饰的请求方法回调将会阻塞当前方法下面的代码执行 , 必须先执行完请求的回调才可以执行下面的代码 , 当然了 await 默认是成功的回调 , 其他回调也可以自己手写
- 同时 , await 阻塞请求方法下面的代码执行会影响到 async 函数下方的代码执行 , 这个先使用了再说
啥也不说了 , 先赶紧学会怎么用 async 和 await 取代 promise 的使用吧
同样是上面的需求
function getP(id = -1) {
console.log(id)
return new Promise((resolve, reject) => {
resolve({data:{id:1}})
})
}
async function get(){
let res = await getP()
getP(res?.data?.id)
}
get()
可以看到 , 第 8 行的 res 确实拿到了值并且传递给下一次请求成功打印出了 1
那如果不加 await 呢?
function getP(id = -1) {
console.log(id)
return new Promise((resolve, reject) => {
resolve({data:{id:1}})
})
}
function get(){
let res = getP()
getP(res?.data?.id)
}
get()
我们可以看到 , 两次的打印结果都是 -1 , 说明第二次并没有值传递进来
好这样子就可以用 async 和 await 替换 promise 发起请求了
await 的作用实际上就是 阻塞代码执行 , 必须先执行完当前这行代码才能走下去 , 相当于将这行代码以下的代码全部转变成了微任务
请先看我得出的两条结论 , 我们将根据这两条结论解读下面的代码逻辑
- 遇到 await 时 , 可以视为将此 await 所在作用域内 await 下方的代码整块全部堵塞 , 堵塞可以视为塞入微任务队列中
- 堵塞的代码中包括的 .then 之内和之后的两部分 , 必须要等待 .then 之内的代码执行完成后 , 才能执行 .then 之后的代码 , 必须要等待 .then 之后的代码执行完成后 , 才能执行堵塞代码所在函数的下一行代码 , 其中需要等待的包括同步任务和微任务 , 而宏任务则统一加入到宏任务队列中
注意 : 对于 await.then 中的嵌套 async 和 await 同样满足上述两条结论 , 而对于 await.then 中的非 await 请求 , 则按照 await.then 中的正常情况处理
请看以下代码为例子:
代码一
// 打印一个 -1 , 然后返回一个 promise 对象并执行成功回调
function getP() {
console.log('-1')
return new Promise((resolve, reject) => {
resolve(1)
})
}
// 创立一个定时器 , 并且马上打印输入的参数
function timer(a) {
setTimeout(() => {
console.log(a)
}, 1);
}
// 声明一个函数 , 主要就是调用这个方法
async function testP() {
console.log('0')
await getP()
.then(async () => {
console.log('1')
timer('2')
await getP()
.then(res => {
console.log('3')
getP()
.then(() => console.log('4'))
console.log('5')
})
console.log('6')
timer('7')
})
console.log('8')
getP()
.then(() => console.log('9'))
console.log('10')
}
// 声明我们要使用的函数 , 其中调用了上述所有的方法
function test() {
timer('11')
testP()
.then(() => {
console.log('12')
})
timer('13')
console.log('同步代码最后一句')
}
// 使用函数
test()
我们先来看下执行结果
接下来我们逐行对上述代码执行过程进行解释 , 请看以下截图 , 因为有行号好讲很多
此时 , 以下没有就是空
打印:
微任务队列:
宏任务队列: 11
此时 , 以下没有就是空
打印: 0
微任务队列:
宏任务队列: 11
此时 , 以下没有就是空
打印: 0 -1
微任务队列: 阻塞代码
宏任务队列: 11
阻塞代码: 第 85 行 到 第 101 行
此时 , 以下没有就是空
打印: 0 -1
微任务队列: 阻塞代码 12
宏任务队列: 11
阻塞代码: 第 85 行 到 第 101 行
此时 , 以下没有就是空
打印: 0 -1
微任务队列: 阻塞代码 12
宏任务队列: 11 13
阻塞代码: 第 85 行 到 第 101 行
此时 , 以下没有就是空
打印: 0 -1 ‘同步代码最后一句’
微任务队列: 阻塞代码 12
宏任务队列: 11 13
阻塞代码: 第 85 行 到 第 101 行
此时 , 以下没有就是空
打印: 0 -1 ‘同步代码最后一句’ 1 -1
微任务队列: 阻塞代码 12
宏任务队列: 11 13 2
阻塞代码: 第 90 行 到 第96 行 , 第 98 行 到 第 101 行
此时 , 以下没有就是空
打印: 0 -1 ‘同步代码最后一句’ 1 -1 3
微任务队列: 阻塞代码 12
宏任务队列: 11 13 2
阻塞代码: 第 90 行 到 第96 行 , 第 98 行 到 第 101 行
此时 , 以下没有就是空
打印: 0 -1 ‘同步代码最后一句’ 1 -1 3 -1 5
微任务队列: 阻塞代码 12 4
宏任务队列: 11 13 2
阻塞代码: 第 98 行 到 第 101 行
此时 , 以下没有就是空
打印: 0 -1 ‘同步代码最后一句’ 1 -1 3 -1 5 4
微任务队列: 阻塞代码 12
宏任务队列: 11 13 2
阻塞代码: 第 98 行 到 第 101 行
此时 , 以下没有就是空
打印: 0 -1 ‘同步代码最后一句’ 1 -1 3 -1 5 4 6
微任务队列: 阻塞代码 12
宏任务队列: 11 13 2 7
阻塞代码: 第 98 行 到 第 101 行
此时 , 以下没有就是空
打印: 0 -1 ‘同步代码最后一句’ 1 -1 3 -1 5 4 6 8 -1
微任务队列: 阻塞代码 12 9
宏任务队列: 11 13 2 7
阻塞代码: 第 98 行 到 第 101 行
此时 , 以下没有就是空
打印: 0 -1 ‘同步代码最后一句’ 1 -1 3 -1 5 4 6 8 -1 10 9
微任务队列: 12
宏任务队列: 11 13 2 7
阻塞代码:
此时 , 以下没有就是空
打印: 0 -1 ‘同步代码最后一句’ 1 -1 3 -1 5 4 6 8 -1 10 9 12
微任务队列:
宏任务队列: 11 13 2 7
阻塞代码:
此时 , 以下没有就是空
打印: 0 -1 ‘同步代码最后一句’ 1 -1 3 -1 5 4 6 8 -1 10 9 12 11 13 2 7
微任务队列:
宏任务队列:
阻塞代码:
好了 , 我们再回顾一次得出的结论
- 遇到 await 时 , 可以视为将此 await 所在作用域内 await 下方的代码整块全部堵塞 , 堵塞可以视为塞入微任务队列中
- 堵塞的代码中包括的 .then 之内和之后的两部分 , 必须要等待 .then 之内的代码执行完成后 , 才能执行 .then 之后的代码 , 必须要等待 .then 之后的代码执行完成后 , 才能执行堵塞代码所在函数的下一行代码 , 其中需要等待的包括同步任务和微任务 , 而宏任务则统一加入到宏任务队列中
注意 : 对于 await.then 中的嵌套 async 和 await 同样满足上述两条结论 , 而对于 await.then 中的非 await 请求 , 则按照 await.then 中的正常情况处理
在 代码一 的基础上我们这次不使用 await 进行修饰 , 所以请看以下代码
代码二
我们做的修改仅仅是将 testP 函数中的 async 和 await 删掉了 , 保留开始的 async 是为了能返回一个 promise 对象
// 打印一个 -1 , 然后返回一个 promise 对象并执行成功回调
function getP() {
console.log('-1')
return new Promise((resolve, reject) => {
resolve(1)
})
}
// 创立一个定时器 , 并且马上打印输入的参数
function timer(a) {
setTimeout(() => {
console.log(a)
}, 1);
}
// 声明一个函数 , 主要就是调用这个方法
async function testP() {
console.log('0')
getP()
.then(() => {
console.log('1')
timer('2')
getP()
.then(res => {
console.log('3')
getP()
.then(() => console.log('4'))
console.log('5')
})
console.log('6')
timer('7')
})
console.log('8')
getP()
.then(() => console.log('9'))
console.log('10')
}
// 声明我们要使用的函数 , 其中调用了上述所有的方法
function test() {
timer('11')
testP()
.then(() => {
console.log('12')
})
timer('13')
console.log('同步代码最后一句')
}
// 使用函数
test()
我们先来看看执行结果是什么
这个其实很简单的了 , 里面可能让人纠结的一点就是微任务队列中再次出现微任务时 , 新的微任务放在哪里 , 答案就是新的微任务放在旧的微任务队列中的最后面 , 这就是为什么执行顺序是 3 会在 12 的后面而不是在 9 的前面
- 遇到 await 时 , 可以视为将此 await 所在作用域内 await 下方的代码整块全部堵塞 , 堵塞可以视为塞入微任务队列中
- 堵塞的代码中包括的 .then 之内和之后的两部分 , 必须要等待 .then 之内的代码执行完成后 , 才能执行 .then 之后的代码 , 必须要等待 .then 之后的代码执行完成后 , 才能执行堵塞代码所在函数的下一行代码 , 其中需要等待的包括同步任务和微任务 , 而宏任务则统一加入到宏任务队列中
注意 : 对于 await.then 中的嵌套 async 和 await 同样满足上述两条结论 , 而对于 await.then 中的非 await 请求 , 则按照 await.then 中的正常情况处理
其实写的比较复杂 , 而且结论我也不敢保证肯定对 , 但是目前就是这样用着
其实你完全不用在乎这些 , 因为除了面试题以外也很少会让你写这样的语句 , 而且你多半也不会这样写
有什么问题赶紧发评论留言大家一起讨论讨论啊
你可以复制下面的 HTML 代码在浏览器的控制台中进行测试
DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>异步title>
<style>
button {
padding: 10px;
}
style>
head>
<body>
<button onclick="queryPromise()">queryPromisebutton>
<pre>
async function queryPromise() {
let obj1
let obj2
obj1 = await getPromise()
obj2 = await getPromise(obj1.id)
console.log('obj1', obj1)
console.log('obj2', obj2)
}
function getPromise(id = 20) {
return new Promise((resolve, reject) => {
let obj
if (id == 20) {
obj = {
id: 12,
name: '张三'
}
} else if (id == 12) {
obj = {
id: 20,
name: '邵雅虎'
}
} else {
obj = {
id: -1,
name: '查无此人'
}
}
resolve(obj)
})
}
pre>
<hr>
<button onclick="queryP()">queryPbutton>
<pre>
async function queryP() {
await getP()
console.log(1)
timer(2)
getP()
.then(() => {
timer(4)
console.log(3)
})
console.log(5)
timer(6)
return new Promise((resolve, reject) => {
resolve('结束')
})
}
function getP() {
console.log('-1')
return new Promise((resolve, reject) => {
resolve(1)
})
}
function timer(a) {
setTimeout(() => {
console.log(a)
}, 1);
}
pre>
<hr>
<button onclick="test()">testbutton>
<pre>
function getP() {
console.log('-1')
return new Promise((resolve, reject) => {
resolve(1)
})
}
function timer(a) {
setTimeout(() => {
console.log(a)
}, 1);
}
async function testP() {
console.log('0')
await getP()
.then(async () => {
console.log('1')
timer('2')
await getP()
.then(res => {
console.log('3')
getP()
.then(() => console.log('4'))
console.log('5')
})
console.log('6')
timer('7')
})
console.log('8')
getP()
.then(() => console.log('9'))
console.log('10')
}
function test() {
timer('11')
testP()
.then(() => {
console.log('12')
})
timer('13')
console.log('同步代码最后一句')
}
pre>
body>
html>
<script>
async function queryPromise() {
let obj1
let obj2
obj1 = await getPromise()
obj2 = await getPromise(obj1.id)
console.log('obj1', obj1)
console.log('obj2', obj2)
}
function getPromise(id = 20) {
return new Promise((resolve, reject) => {
let obj
if (id == 20) {
obj = {
id: 12,
name: '张三'
}
} else if (id == 12) {
obj = {
id: 20,
name: '邵雅虎'
}
} else {
obj = {
id: -1,
name: '查无此人'
}
}
resolve(obj)
})
}
async function queryP() {
await getP()
console.log(1)
timer(2)
getP()
.then(() => {
timer(4)
console.log(3)
})
console.log(5)
timer(6)
return new Promise((resolve, reject) => {
resolve('结束')
})
}
function getP() {
console.log('-1')
return new Promise((resolve, reject) => {
resolve(1)
})
}
function timer(a) {
setTimeout(() => {
console.log(a)
}, 1);
}
async function testP() {
console.log('0')
await getP()
.then(async () => {
console.log('1')
timer('2')
await getP()
.then(res => {
console.log('3')
getP()
.then(() => console.log('4'))
console.log('5')
})
console.log('6')
timer('7')
})
console.log('8')
getP()
.then(() => console.log('9'))
console.log('10')
}
function test() {
timer('11')
testP()
.then(() => {
console.log('12')
})
timer('13')
console.log('同步代码最后一句')
}
script>