Javascript中的async awai的用法
async / await是es7的重要特性之一,也是目前社区里公认的优秀异步解决方案。目前,async / await这个特性已经是stage 3的建议,可以看看tc39的进度,本篇文章将分享async / await是如何工作的,阅读本文前,希望你具备promise、generator、yield等es6的相关知识。
在详细介绍async / await之前,先回顾下目前在es6中比较好的异步处理办法。下面的例子中数据请求用node.js中的request模块,数据接口采用github v3 api文档提供的repo代码仓库详情api作为例子演示。
promise对异步的处理
虽然node.js的异步io带来了对高并发的良好支持,同时也让“回调”成为灾难,很容易造成回调地狱。传统的方式比如使用具名函数,虽然可以减少嵌套的层数,让代码看起来比较清晰。但是会造成比较差的编码和调试体验,你需要经常使用用ctrl + f去寻找某个具名函数的定义,这使得ide窗口经常上下来回跳动。使用promise之后,可以很好的减少嵌套的层数。另外promise的实现采用了状态机,在函数里面可以很好的通过resolve和reject进行流程控制,你可以按照顺序链式的去执行一系列代码逻辑了。下面是使用promise的一个例子:
const request = require('request'); // 请求的url和header const options = { url: 'https://api.github.com/repos/cpselvis/zhihu-crawler', headers: { 'user-agent': 'request' } }; // 获取仓库信息 const getrepodata = () => { return new promise((resolve, reject) => { request(options, (err, res, body) => { if (err) { reject(err); } resolve(body); }); }); }; getrepodata() .then((result) => console.log(result);) .catch((reason) => console.error(reason);); // 此处如果是多个promise顺序执行的话,如下: // 每个then里面去执行下一个promise // getrepodata() // .then((value2) => {return promise2}) // .then((value3) => {return promise3}) // .then((x) => console.log(x))
不过promise仍然存在缺陷,它只是减少了嵌套,并不能完全消除嵌套。举个例子,对于多个promise串行执行的情况,第一个promise的逻辑执行完之后,我们需要在它的then函数里面去执行第二个promise,这个时候会产生一层嵌套。另外,采用promise的代码看起来依然是异步的,如果写的代码如果能够变成同步该多好啊!
generator对异步的处理
谈到generator,你应该不会对它感到陌生。在node.js中对于回调的处理,我们经常用的tj / co就是使用generator结合promise来实现的,co是coroutine的简称,借鉴于python、lua等语言中的协程。它可以将异步的代码逻辑写成同步的方式,这使得代码的阅读和组织变得更加清晰,也便于调试。
const co = require('co'); const request = require('request'); const options = { url: 'https://api.github.com/repos/cpselvis/zhihu-crawler', headers: { 'user-agent': 'request' } }; // yield后面是一个生成器 generator const getrepodata = function* () { return new promise((resolve, reject) => { request(options, (err, res, body) => { if (err) { reject(err); } resolve(body); }); }); }; co(function* () { const result = yield getrepodata; // ... 如果有多个异步流程,可以放在这里,比如 // const r1 = yield getr1; // const r2 = yield getr2; // const r3 = yield getr3; // 每个yield相当于暂停,执行yield之后会等待它后面的generator返回值之后再执行后面其它的yield逻辑。 return result; }).then(function (value) { console.log(value); }, function (err) { console.error(err.stack); });
async / await对异步的处理
虽然co是社区里面的优秀异步解决方案,但是并不是语言标准,只是一个过渡方案。es7语言层面提供async / await去解决语言层面的难题。目前async / await 在 ie edge中已经可以直接使用了,但是chrome和node.js还没有支持。幸运的是,babel已经支持async的transform了,所以我们使用的时候引入babel就行。在开始之前我们需要引入以下的package,preset-stage-3里就有我们需要的async/await的编译文件。
无论是在browser还是node.js端都需要安装下面的包。
$ npm install babel-core --save $ npm install babel-preset-es2015 --save $ npm install babel-preset-stage-3 --save
这里推荐使用babel官方提供的require hook方法。就是通过require进来后,接下来的文件进行require的时候都会经过babel的处理。因为我们知道commonjs是同步的模块依赖,所以也是可行的方法。这个时候,需要编写两个文件,一个是启动的js文件,另外一个是真正执行程序的js文件。
启动文件index.js
require('babel-core/register'); require('./async.js');
真正执行程序的async.js
const request = require('request'); const options = { url: 'https://api.github.com/repos/cpselvis/zhihu-crawler', headers: { 'user-agent': 'request' } }; const getrepodata = () => { return new promise((resolve, reject) => { request(options, (err, res, body) => { if (err) { reject(err); } resolve(body); }); }); }; async function asyncfun() { try { const value = await getrepodata(); // ... 和上面的yield类似,如果有多个异步流程,可以放在这里,比如 // const r1 = await getr1(); // const r2 = await getr2(); // const r3 = await getr3(); // 每个await相当于暂停,执行await之后会等待它后面的函数(不是generator)返回值之后再执行后面其它的await逻辑。 return value; } catch (err) { console.log(err); } } asyncfun().then(x => console.log(`x: ${x}`)).catch(err => console.error(err));
注意点:
- async用来申明里面包裹的内容可以进行同步的方式执行,await则是进行执行顺序控制,每次执行一个await,程序都会暂停等待await返回值,然后再执行之后的await。
- await后面调用的函数需要返回一个promise,另外这个函数是一个普通的函数即可,而不是generator。
- await只能用在async函数之中,用在普通函数中会报错。
- await命令后面的 promise 对象,运行结果可能是 rejected,所以最好把 await 命令放在 try...catch 代码块中。
其实,async / await的用法和co差不多,await和yield都是表示暂停,外面包裹一层async 或者 co来表示里面的代码可以采用同步的方式进行处理。不过async / await里面的await后面跟着的函数不需要额外处理,co是需要将它写成一个generator的。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
推荐阅读
-
JavaScript中this的全面解析及常见实例
-
JavaScript在web自动化测试中的作用示例详解
-
从源码解析Python的Flask框架中request对象的用法
-
C#中datagridview的EditingControlShowing事件用法实例
-
Android中的Selector的用法详解及实例
-
详解Python3中yield生成器的用法
-
python中的lambda表达式用法详解
-
Java多线程编程中ThreadLocal类的用法及深入
-
对比Python中__getattr__和 __getattribute__获取属性的用法
-
深入理解Javascript中的valueOf与toString