理解 async/await
ES7 提出的 async
函数,终于让 JavaScript 对于异步操作有了终极解决方案。No more callback hell。async
函数是 Generator
函数的语法糖。使用 关键字 async
来表示,在函数内部使用 await
来表示异步。
想较于 Generator,Async
函数的改进在于下面四点:
- 内置执行器。Generator 函数的执行必须依靠执行器,而
Aysnc
函数自带执行器,调用方式跟普通函数的调用一样 - 更好的语义。
async
和await
相较于*
和yield
更加语义化 - 更广的适用性。
co
模块约定,yield
命令后面只能是 Thunk 函数或 Promise对象。而async
函数的await
命令后面则可以是 Promise 或者 原始类型的值(Number,string,boolean,但这时等同于同步操作) - 返回值是 Promise。
async
函数返回值是 Promise 对象,比 Generator 函数返回的 Iterator 对象方便,可以直接使用then()
方法进行调用
Async 与其他异步操作的对比
先定义一个 Fetch 方法用于获取 github user 的信息:
1 | function fetchUser() { |
Promise 方式
1 | /** |
Promise 的方式虽然解决了 callback hell,但是这种方式充满了 Promise的 then()
方法,如果处理流程复杂的话,整段代码将充满 then
。语义化不明显,代码流程不能很好的表示执行流程。
Generator 方式
1 | /** |
Generator 的方式解决了 Promise 的一些问题,流程更加直观、语义化。但是 Generator 的问题在于,函数的执行需要依靠执行器,每次都需要通过 g.next()
的方式去执行。
async 方式
1 | /** |
async
函数完美的解决了上面两种方式的问题。流程清晰,直观、语义明显。操作异步流程就如同操作同步流程。同时 async
函数自带执行器,执行的时候无需手动加载。
语法
async 函数返回一个 Promise 对象
async
函数内部 return 返回的值。会成为 then
方法回调函数的参数。
1 | async function f() { |
如果 async
函数内部抛出异常,则会导致返回的 Promise 对象状态变为 reject
状态。抛出的错误而会被 catch
方法回调函数接收到。
1 | async function e(){ |
async 函数返回的 Promise 对象,必须等到内部所有的 await 命令的 Promise 对象执行完,才会发生状态改变
也就是说,只有当 async
函数内部的异步操作都执行完,才会执行 then
方法的回调。
1 | const delay = timeout => new Promise(resolve=> setTimeout(resolve, timeout)); |
正常情况下,await 命令后面跟着的是 Promise ,如果不是的话,也会被转换成一个 立即 resolve 的 Promise
如下面这个例子:
1 | async function f() { |
如果返回的是 reject 的状态,则会被 catch
方法捕获。
Async 函数的错误处理
async
函数的语法不难,难在错误处理上。
先来看下面的例子:
1 | let a; |
如上面所示,当 async
函数中只要一个 await
出现 reject 状态,则后面的 await
都不会被执行。
解决办法:可以添加 try/catch
。
1 | // 正确的写法 |
如果有多个 await
则可以将其都放在 try/catch
中。
如何在项目中使用
依然是通过 babel
来使用。
只需要设置 presets
为 stage-3
即可。
安装依赖:1
npm install babel-preset-es2015 babel-preset-stage-3 babel-runtime babel-plugin-transform-runtime
修改.babelrc
:
1 | "presets": ["es2015", "stage-3"], |
这样就可以在项目中使用 async
函数了。