async/await 是ES8规范新增的,使得以同步方式写的代码异步运行不再是白日梦,进一步让代码逻辑更加清晰。
下面有这样一个需求:有两个请求,请求 1 的结果是请求 2 的参数,所以请求 2 必须在请求 1 之后发出,使用 Promise 实现如下:
request('/xxx').then((res) =>{
request('/yyyy',{res}).then(() => {
// 对结果进行处理
})
})
可以看到这里形成了一个嵌套结构,试想如果这样的嵌套层级深了以后就会形成类似“回调地狱”那样的结构,很不利于维护。因此就新增了 async/await 这两个关键字来解决这个问题。上面的需求 async/await 写法如下:
async function fn() {
let res = await request('/xxx');
request('/yyyy',{res}).then(() => {
// 对结果进行处理
})
}
async 关键字写在函数名之前(相当于一个标识),让普通函数具有异步的行为特征,但是整体上代码却是同步执行的。它可以用在以下地方:
async function fn1() { } //函数声明
let fn2 = async function () { }//函数表达式
let fn3 = async () => { }//箭头函数
let test = { // 类中属性方法
async fn4() { }
}
// 总体同步执行
async function fn5() {
console.log(1);
}
fn5();// 1
console.log(2);//2
异步函数返回值是一个期约对象,内部逻辑是使用 Promise.resolve方法来处理返回值 。因此可以使用.then方法处理异步任务的结果。
async function fn(){
console.log('1.异步函数执行了');
const res = await '2.插队?不讲武德';
}
fn().then(res => console.log(res));
async function fn(){
console.log('异步函数执行了');
}
const res = fn();
console.log(res);

async function fn(){
console.log('异步函数执行了');
return '我是返回值';
}
const res = fn();
console.log(res);

async function fn(){
console.log('异步函数执行了');
throw new Error('出错了');
}
const res = fn();
console.log(res);

async function fn(){
console.log('异步函数执行了');
return Promise.reject('出错了');
}
const res = fn();
console.log(res);

当遇到 await 后,会先暂停异步函数的执行,让出 JS 运行时线程,执行别的代码。当异步函数拿到结果又会在合适的时机恢复运行。
async function fn(){
console.log('1.异步函数执行了');
const res = await '2.插队?不讲武德';
console.log(res);
console.log('3.终于轮到我了')
}
fn();
console.log('4.我先插个队');

注意:await 关键字只会暂停异步函数的执行,并不会影响异步函数之外的代码。如上面的代码执行 fn 函数打印“1.异步函数执行了”,遇到 await 关键字,就暂停函数中 await 后边的代码执行,转而去打印“4.我先插个队”。
早期 await 只允许在异步函数中使用,再同步函数使用会报错。

但是从 ES2022 开始,允许在模块的顶层独立使用await命令。如下所示:

它的主要目的是使用await解决模块异步加载的问题。
因为 await 等待的是一个异步操作,而异步任务有可能失败(rejected),所以需要把 await 放在 try/catch中。
function fn() {
try{
let res = await ···;
let res1 = await ···;
let res2 = await ···;
} catch(err) {
// 失败处理逻辑
}
}
async function async1() {
console.log("async1 1");
await async2();
console.log("async1 2");
setTimeout(() => {
console.log('timeout1')
}, 0)
}
async function async2() {
setTimeout(() => {
console.log('timeout2')
}, 0)
console.log("async2");
}
async1();
setTimeout(() => {
console.log('timeout3')
}, 0)
console.log("start")

async1函数,打印“async1 1”,遇到await,暂停async1函数执行,向消息队列添加一个在async2执行完之后执行的任务;async2函数,遇到setTimeout,将其添加到宏任务队列,打印 “async2”,async2函数执行完毕,把给 await 提供值的任务添加到消息队列,async1退出;setTimeout,将其添加到宏任务队列,打印“start”,同步代码执行完毕。async1函数执行的任务。async1函数执行的任务,打印“async1 2”;遇到setTimeout添加到宏任务队列。分析上面的代码我发现 await 并没有等待 async2 函数中 setTimeout 函数执行结束,原因是 setTimeout 函数在调用时会同步返回一个随机数,而实际上 await 等待的就是异步任务结果,此时,这个随机数就会被await 认为是 setTimeout 异步任务的结果,就放行了。
async function fn() {
await fn1();
await fn2();
}
async function fn1() {
setTimeout(() => {
// 省略代码
}, 200);
}
async function fn2() {
Math.random();
}
上面的代码 fn1 和 fn2 函数对于 await 来说是等价的。
生成器是 ES6 新增的,它拥有在一个函数块内暂停和恢复代码执行的能力。(箭头函数不能用来定义生成器函数)只要调用生成器函数就会产生一个生成器对象,它实现了 Iterator 接口,初始为暂停执行状态,可以调用 next 方法让生成器恢复执行。
next 方法的返回值类似于迭代器.{value:xxx,done:flase||true}当done为true 就意味着生成器已经执行完毕,此时再调用next方法,value是生成器函数的返回值,默认为 undefined。
function* fn() {
yield 1;
yield 2;
}
// 生成生成器对象,暂停状态
const Fn = fn();
// 恢复执行
console.log(Fn.next());//{value: 1, done: false}
// 恢复执行
console.log(Fn.next());//{value: 2, done: false}
// 执行完毕
console.log(Fn.next());//{value: undefined, done: true}
// 执行完毕后再调用 next 函数,会输出同样的结果
console.log(Fn.next());//{value: undefined, done: true}
yield 关键字可以让生成器停止执行。生成器函数在遇到 yield 关键字后会暂停执行并保留函数作用域状态,只能通过调用生成器对象的 next 方法恢复执行。并且 yield 语句生成的值就在 next 函数返回值里,此时生成器函数时done:false的状态,通过 return 退出生成器会处于done:true状态
function* fn() {
yield 1;
return 2;
}
const Fn = fn();
// yield 生成的值就是 next 函数返回的对象里的 value
console.log(Fn.next());//{value: 1, done: false}
console.log(Fn.next());//{value: 2, done: true}
因为生成器对象实现了 Iterator 接口,可以使用 for of 遍历,它会自动调用 next 方法。
function* fn() {
yield 1;
yield 2;
yield 3;
}
for (let item of fn()) {
console.log(item);
}

生成器对象的 next 方法是可以接收一个参数的,这个参数最终会传给 yield ,注意两点:
console.log('开始执行');,所以这是传参没有 yield 接收。function* fn() {
console.log('开始执行');
let r1 = yield 1;
let r2 = yield r1;
return r2;
}
const Fn = fn();
console.log(Fn.next());//{value: 1, done: false}
console.log(Fn.next(2));//{value: 2, done: false}
console.log(Fn.next(3));//{value: 3, done: true}
return 会强制生成器进入关闭状态,不可逆,最后的状态值就是传给return()的参数。并且调用return以后再调用next方法都会返回{value:undefined,done:true};
function* fn() {
yield 1;
return 2;
yield 3;
}
const Fn = fn();
console.log(Fn.next());//{value: 1, done: false}
console.log(Fn.next());//{value: 2, done: true}
console.log(Fn.next());//{value: undefined, done: true}
console.log(Fn.next());//{value: undefined, done: true}
throw 方法会在暂停的时候将一个错误注入到生成器对象中,如果错误未被处理,生成器就会关闭。
function* fn() {
yield 1;
yield 2;
yield 3;
}
const Fn = fn();
console.log(Fn.next());//{value: 1, done: false}
Fn.throw('出错了···')
console.log(Fn.next());
console.log(Fn.next());

如果生成器函数内部处理了这个错误,只会跳过对应的yield,可再次恢复执行。
function* fn() {
try {
yield 1;
} catch (error) {
console.log('捕获到错误内容:'+ error)
}
yield 2;
yield 3;
}
const Fn = fn();
console.log(Fn.next());//{value: 1, done: false}
Fn.throw('出错了···');
console.log(Fn.next());//{value: 3, done: false}
console.log(Fn.next());//{value: undefined, done: true}

还记得在介绍 async/await 时说过的需求吗?请求一的结果是请求二的参数,最后拿到结果,下面利用生成器和 Promise 来模拟,注意这样的代码就类似于 async/await 是同步的逻辑。
function fn(x) {
return new Promise((resolve,reject) => {
setTimeout(() => {
resolve(x + 1);
}, 1000);
})
}
function* fn1() {
console.log('开始执行');
let r1 = yield fn(1);
console.log('请求一完成');// 1s后打印
let r2 = yield fn(r1);
console.log('请求二完成');// 2s后打印
return r2;
}
const gen = fn1();
gen.next().value.then((res) => {
gen.next(res).value.then(res => {
console.log('输出结果', gen.next(res));
})
})
上面的代码有很多缺陷,只能执行有限步···,现在我们可以稍稍封装一下
function fn(x) {
return new Promise((resolve,reject) => {
setTimeout(() => {
resolve(x + 1);
}, 1000);
})
}
function* fn1() {
console.log('开始执行');
let r1 = yield fn(1);
console.log('请求一完成');
let r2 = yield fn(r1);
console.log('请求二完成');
return r2;
}
function myAsync (fn) {
const Fn = fn.apply(this, arguments);//拿到生成器对象
return new Promise((resolve,reject) => {
function forward(key,val) {
let res = null;
try{
res = Fn[key](val);//恢复执行
} catch(err) {
return reject(err);
}
let {value,done} = res;
if (done) {//代码执行完了,返回一个解决状态的期约
return resolve(value);
} else {
return Promise.resolve(value).then(value => forward('next',value),err=>forward('throw',err));
}
}
forward('next');//第一次执行
})
}
const asyncFn = myAsync(fn1)
asyncFn.then(res => console.log(res))
这样 fn1 函数中 yield 就相当于 await ,且异步函数 myAsync 执行完就能得到一个期约,并且无论 fn1 函数中有多少 yield 都能正确执行,并得到结果。
async/await 和 生成的使用细节以及具体的应用还有很多,我在这里只是稍稍总结一下,本篇博客内容也是我的学习笔记吧。我是孤城浪人,一名正在前端路上摸爬滚打的菜鸟,欢迎你的关注。