async
函数的返回值是Promise
对象
怎么理解上面这句话呢?我们知道,普通的函数的执行结果要有返回值的话,需要用return
关键字,没有return
关键字则返回值是 undefined
const f = () => {
doSomething()...
}
const res1 = f() // undefined
const g = () => {
doSomething()...
return 'hello world'
}
const res2 = g() // 'hello world'
但是async
函数不一样,async
函数一定有返回值,返回值是一个promise 对象
;这是怎么做到的呢?async
函数中的 return
关键字有什么用呢?
-
如果
async
函数中没有return
语句,async
函数会自动返回Promise.resolve(undefined)
,于是就返回了一个Promise对象
-
如果
async
函数return
一个直接量xxx,async
函数会自动返回Promise.resolve(xxx)
,于是返回的Promise对象
的状态是resolved
。在调用async
函数后,接收return
返回的值只能用then
方法。
async function testAsync() {
const res = await fetch('xxx') //返回`hello async`
doSomething()...
return res;
}
testAsync().then(v => {
console.log(v); // 输出 hello async
});
console.log('123')
async
函数内部,await
后面如果是一个 Promise 对象,它会阻塞后面的代码,等着 Promise 对象 resolve,然后得到 resolve 的值,作为 await 表达式的运算结果,然后继续向下执行。
async
函数外部,调用async
函数后,无论里面是否有耗时操作,都会立即返回,继续向下执行。所以,上面代码,会先输出123,再输出hello async
async函数返回的 Promise 对象,必须等到内部所有await命令后面的 Promise 对象执行完,才会发生状态改变,除非遇到return语句或者抛出错误
async函数内部抛出错误,会导致返回的 Promise 对象变为reject状态。抛出的错误对象会被外面添加的catch
方法回调函数接收到。
任何一个await语句后面的 Promise 对象变为reject状态,那么整个async函数都会中断执行。
如果我们希望即使前一个异步操作失败,也不要中断后面的异步操作。这时可以将第一个await放在try...catch结构里面,这样不管这个异步操作是否成功,第二个await都会执行。(ps:因为async函数里面有try catch
捕获了异常,没有抛到外面来,所以外面async函数返回的promise不会被reject)
如果await后面的异步操作出错,又没有在async
函数内被我们主动catch
,那么async函数返回的 Promise 对象就会被reject。
最佳实践
async await
本来就是来替代 promise ...then...
的,写成同步的语法,比一连串的then
简洁很多,但是,async
返回的还是一个promise
对象,怎么处理这里的矛盾呢?
我的建议是,在async函数中,加入try catch
,所有的逻辑写在try catch
中,并且一般情况下不需要return xxx,直接把后续逻辑在async
函数中写完,而不是return xxx后,在 then
中写后续逻辑;不过这样一来,发生错误后,被里面的catch
捕获了,返回的promise仍然是resolved
,如果要在外层catch
中处理错误,而不是在async中处理错误,可以在里面catch
到错误后,调用Promise.reject()
来手动让外层catch到错误。
show me code :
const f1 = async () => {
console.log('第一个function start');
const res = await new Promise((resolve) => {
setTimeout(() => {
console.log('第一个async');
resolve('f1 resolved');
}, 3000);
});
console.log('第一个function end');
return res;
};
const f2 = async () => {
console.log('第二个function start');
const res = await new Promise((resolve) => {
setTimeout(() => {
console.log('第二个async');
resolve('f2 resolved');
}, 3000);
});
console.log('第二个function end');
return res;
};
const test1 = () => {
const res1 = f1();
const res2 = f2();
console.log(123456789, res1, res2);
};
test1();
// output:
// 同时:
// 第一个function start
// 第二个function start
// 123456789, 两个 promise 对象
// 3s后 同时:
// 第一个async
// 第二个async
// 第一个function end
// 第二个function end
// 这里 `f1` 和 `f2` 是同步执行的
const test2 = () => {
f1().then(res => {console.log(res);});
f2().then(res => {console.log(res);});
console.log(123456789);
};
test2();
// output:
// 同时:
// 第一个function start
// 第二个function start
// 123456789, 两个 promise 对象
// 3s后 同时:
// 第一个async
// 第二个async
// 第一个function end
// 第二个function end
// f1 resolved
// f2 resolved
// 和上面一样 这里 `f1` 和 `f2` 是同步执行的
const test3 = async () => {
const res1 = await f1();
const res2 = await f2();
console.log(123456789, res1, res2);
};
test3();
// output:
// 立即
// 第一个function start
// 3s 后 同时打印:
// 第一个async
// 第一个function end
// 第二个function start
// 再3s后
// 第二个async
// 第二个function end
// 123456789 "f1 resolved" "f2 resolved"
总结: async
函数抽出后,要想执行 async
函数时是等待阻塞的,还得在外面嵌套async函数, 因为 async
函数返回的还是 Promise
对象
网友评论