ホームページ >ウェブフロントエンド >jsチュートリアル >ES7のasync/await使用例を詳しく解説
この記事は主に ES7 での async/await の使い方を詳しく紹介しています。編集者が非常に優れていると考えたので、参考として共有します。エディターをフォローして見てみましょう
ES6 Promise について初めて知ったとき、私は「JS コードの非同期処理を解決するためのジェネレーター関数を使用した Promise と Co の比較」というブログ記事を書きました。その記事では、Promise と Co の使用を比較しました。 co モジュールはジェネレーター関数と組み合わされて、js 非同期の類似点と相違点を解決します。
記事の最後で、ES7 の async と await について言及しましたが、当時は簡単に言及されただけで、深くは議論されていませんでした。
2 か月前にリリースされた Nodejs V7 では、async と await のサポートが追加されました。今日は、これについて詳しく見てみましょう。非同期コードをよりエレガントな方法で記述します。
async/await とは何ですか?
async/await は、co モジュールとジェネレーター関数の糖衣構文であると言えます。より明確なセマンティクスで js 非同期コードを解決します。
co モジュールに精通している学生は、co モジュールがマスター TJ によって作成されたモジュールであり、ジェネレーター関数を使用して非同期プロセスを解決するモジュールであることを知っているはずです。ジェネレーター関数の実行者と見なすことができます。 Async/await は co モジュールへのアップグレードであり、ジェネレーター関数エグゼキューターが組み込まれており、co モジュールに依存しなくなりました。同時に、async は Promise を返します。
上記の観点から、co モジュールにしろ async/await にしろ、Promise についてよく知らない学生は、まず Promise について詳しく学ぶことができます。
Promise、co、async/await の比較
簡単な例を使用して、3 つのメソッド間の類似点、相違点、トレードオフを比較してみましょう。
例として、mongodb データベースのクエリに mongodb の nodejs ドライバーを使用します。これは、mongodb の js ドライバーがデフォルトで Promise を返すように実装されており、Promise を個別にラップする必要がないためです。
Promise チェーンを使用する
MongoClient.connect(url + db_name).then(db=> { return db.collection('blogs'); }).then(coll=> { return coll.find().toArray(); }).then(blogs=> { console.log(blogs.length); }).catch(err=> { console.log(err); })
Promise の then() メソッドは、別の Promise または同期された値を返すことができます。同期された値が返された場合、それは Promise にパッケージ化されます。
上記の例では、db.collection() は同期された値、つまりコレクション オブジェクトを返しますが、それは Promise にラップされ、次の then() メソッドに透過的に渡されます。
上記の例では Promise チェーンを使用しています。
まずデータベースに接続します MongoClient.connect() は Promise を返し、次に then() メソッドでデータベース オブジェクト db を取得し、次に coll オブジェクトを取得して返します。次の then() メソッドで coll オブジェクトを取得し、クエリを実行してクエリ結果を返し、then() メソッドをレイヤーごとに呼び出して Promise チェーンを形成します。
この Promise チェーンでは、いずれかのリンクで例外が発生すると、最後の catch() によってキャッチされます。
Promise チェーンを使用して記述されたこのコードは、レイヤーごとにコールバック関数を呼び出すよりもエレガントで、プロセスが明確であると言えます。最初にデータベース オブジェクトを取得し、次にコレクション オブジェクトを取得し、最後にデータをクエリします。
しかし、ここにはあまり「エレガント」ではない問題があります。それは、各 then() メソッドによって取得されたオブジェクトが、前の then() メソッドによって返されたデータであるということです。レイヤーを越えてアクセスすることはできません。
これはどういう意味ですか? 3 番目の then (blogs => {}) では、クエリ結果 blogs のみを取得できますが、上記の db オブジェクトと coll オブジェクトは使用できないことを意味します。このとき、ブログ一覧を出力した後にデータベース db.close() を閉じたい場合はどうすればよいでしょうか。
現時点では、解決策が 2 つあります:
1 つ目は、then() ネストを使用することです。コールバック関数を使用するのと同じように、Promise チェーンを中断してネストさせます。
MongoClient.connect(url + db_name).then(db=> { let coll = db.collection('blogs'); coll.find().toArray().then(blogs=> { console.log(blogs.length); db.close(); }).catch(err=> { console.log(err); }); }).catch(err=> { console.log(err); })
ここでは、最後のクエリ操作で外部データベースをオブジェクトと呼び出せるように、2 つの Promise をネストします。ただし、この方法はお勧めできません。理由は簡単です。ある種類のコールバック関数地獄から別の種類の Promise コールバック地獄に変わったからです。
さらに、Promise はチェーンを形成していないため、各 Promise の例外をキャッチする必要があります。
別の方法があります。それは、各 then() メソッドで db を渡すことです:
MongoClient.connect(url + db_name).then(db=> { return {db:db,coll:db.collection('blogs')}; }).then(result=> { return {db:result.db,blogs:result.coll.find().toArray()}; }).then(result=> { return result.blogs.then(blogs=> { //注意这里,result.coll.find().toArray()返回的是一个Promise,因此这里需要再解析一层 return {db:result.db,blogs:blogs} }) }).then(result=> { console.log(result.blogs.length); result.db.close(); }).catch(err=> { console.log(err); });
各 then() メソッドの戻りでは、毎回 db とその他の結果をオブジェクトとして返します。 。各結果が同期された値である場合は問題ありませんが、それが Promise 値である場合、各 Promise には追加の解析層が必要になることに注意してください。
たとえば、上記の例では、2 番目の then() メソッドによって返される {db:result.db,blogs:result.coll.find().toArray()} オブジェクトで、blogs は Promise であり、次の then() ) メソッドでは、ブログ リストの配列値を直接参照できないため、 then() メソッドを呼び出して最初に 1 つのレイヤーを解析し、次に 2 つの同期された値 db と blogs を返す必要があります。
これには Promise のネストが含まれますが、Promise は then() の 1 レベルのみをネストすることに注意してください。
このメソッドは非常に面倒なメソッドでもあります。then() メソッドが同期された値ではなく Promise を返す場合、さらに多くの作業を行う必要があるからです。さらに、毎回「追加の」db オブジェクトを透過的に送信するのは、論理的に少し冗長です。
但除此之外,对于Promise链的使用,如果遇到上面的问题,好像也没其他更好的方法解决了。我们只能根据场景去选择一种“最优”的方案,如果要使用Promise链的话。
鉴于Promise上面蛋疼的问题,TJ大神将ES6中的生成器函数,用co模块包装了一下,以更优雅的方式来解决上面的问题。
co搭配生成器函数
如果使用co模块搭配生成器函数,那么上面的例子可以改写如下:
const co = require('co'); co(function* (){ let db = yield MongoClient.connect(url + db_name); let coll = db.collection('blogs'); let blogs = yield coll.find().toArray(); console.log(blogs.length); db.close(); }).catch(err=> { console.log(err); });
co是一个函数,将接受一个生成器函数作为参数,去执行这个生成器函数。生成器函数中使用 yield 关键字来“同步”获取每个异步操作的值。
上面代码在代码形式上,比上面使用Promise链要优雅,我们消灭了回调函数,代码看起来都是同步的。除了使用co和yield有点怪之外。
使用co模块,我们要将所有的操作包装成一个生成器函数,然后使用co()去调用这个生成器函数。看上去也还可以接受,但是ES的进化是不满足于此的,于是async/await被提到了ES7的提案。
async/await
我们先看一下使用async/await改写上面的代码:
(async function(){ let db = await MongoClient.connect(url + db_name); let coll = db.collection('blogs'); let blogs = await coll.find().toArray(); console.log(blogs.length); db.close(); })().catch(err=> { console.log(err); });
我们对比代码可以看出,async/await和co两种方式代码极为相似。
co换成了async,yield换成了await。同时生成器函数变成了普通函数。
这种方式在语义上更加清晰明了,async表明这个函数是异步的,同时await表示要“等待”异步操作返回值。
async函数返回一个Promise,上面的代码其实是这样:
let getBlogs = async function(){ let db = await MongoClient.connect(url + db_name); let coll = db.collection('blogs'); let blogs = await coll.find().toArray(); db.close(); return blogs; }; getBlogs().then(result=> { console.log(result.length); }).catch(err=> { console.log(err); })
我们定义getBlogs为一个async函数,最后返回得到的博客列表最终会被包装成一个Promise返回,如上,我们直接调用getBlogs().then()方法可获取async函数返回值。
好了,上面我们简单对比了一下三种解决异步方案,下面我们来深入了解一下async/await。
深入async/await
async返回值
async用于定义一个异步函数,该函数返回一个Promise。
如果async函数返回的是一个同步的值,这个值将被包装成一个理解resolve的Promise,等同于return Promise.resolve(value)
。
await用于一个异步操作之前,表示要“等待”这个异步操作的返回值。await也可以用于一个同步的值。
//返回一个Promise let timer = async functiontimer(){ return new Promise((resolve,reject) => { setTimeout(()=> { resolve('500'); },500); }); } timer().then(result=> { console.log(result); //500 }).catch(err=> { console.log(err.message); });
//返回一个同步的值 let sayHi = async functionsayHi(){ let hi = await 'hello world'; return hi; //等同于return Promise.resolve(hi); } sayHi().then(result=> { console.log(result); });
上面这个例子返回是一个同步的值,字符串'hello world',sayHi()是一个async函数,返回值被包装成一个Promise,可以调用then()方法获取返回值。
对于一个同步的值,可以使用await,也可以不使用await。效果效果是一样的。具体用不用,看情况。
比如上面使用mongodb查询博客那个例子, let coll = db.collection('blogs'); ,这里我们就没有用await,因为这是一个同步的值。当然,也可以使用await,这样会显得代码统一。虽然效果是一样的。
async函数的异常
let sayHi = async functionsayHi(){ throw new Error('出错了'); } sayHi().then(result=> { console.log(result); }).catch(err=> { console.log(err.message); //出错了 });
我们直接在async函数中抛出一个异常,由于返回的是一个Promise,因此,这个异常可以调用返回Promise的catch()方法捕捉到。
和Promise链的对比:
我们的async函数中可以包含多个异步操作,其异常和Promise链有相同之处,如果有一个Promise被reject()那么后面的将不会再进行。
let count = ()=>{ return new Promise((resolve,reject) => { setTimeout(()=>{ reject('故意抛出错误'); },500); }); } let list = ()=>{ return new Promise((resolve,reject)=>{ setTimeout(()=>{ resolve([1,2,3]); },500); }); } let getList = async ()=>{ let c = await count(); let l = await list(); return {count:c,list:l}; } console.time('begin'); getList().then(result=> { console.log(result); }).catch(err=> { console.timeEnd('begin'); console.log(err); }); //begin: 507.490ms //故意抛出错误
如上面的代码,定义两个异步操作,count和list,使用setTimeout延时500毫秒,count故意直接抛出异常,从输出结果来看,count()抛出异常后,直接由catch()捕捉到了,list()并没有继续执行。
并行
使用async后,我们上面的例子都是串行的。比如上个list()和count()的例子,我们可以将这个例子用作分页查询数据的场景。
先查询出数据库中总共有多少条记录,然后再根据分页条件查询分页数据,最后返回分页数据以及分页信息。
我们上面的例子count()和list()有个“先后顺序”,即我们先查的总数,然后又查的列表。其实,这两个操作并无先后关联性,我们可以异步的同时进行查询,然后等到所有结果都返回时再拼装数据即可。
let count = ()=>{ return new Promise((resolve,reject) => { setTimeout(()=>{ resolve(100); },500); }); } let list = ()=>{ return new Promise((resolve,reject)=>{ setTimeout(()=>{ resolve([1,2,3]); },500); }); } let getList = async ()=>{ let result = await Promise.all([count(),list()]); return result; } console.time('begin'); getList().then(result=> { console.timeEnd('begin'); //begin: 505.557ms console.log(result); //[ 100, [ 1, 2, 3 ] ] }).catch(err=> { console.timeEnd('begin'); console.log(err); });
我们将count()和list()使用Promise.all()“同时”执行,这里count()和list()可以看作是“并行”执行的,所耗时间将是两个异步操作中耗时最长的耗时。
最后得到的结果是两个操作的结果组成的数组。我们只需要按照顺序取出数组中的值即可。
JavaScript 中最蛋疼的事情莫过于回调函数嵌套问题。以往在浏览器中,因为与服务器通讯是一种比较昂贵的操作,因此比较复杂的业务逻辑往往都放在服务器端,前端 JavaScript 只需要少数几次 AJAX 请求就可拿到全部数据。
但是到了 webapp 风行的时代,前端业务逻辑越来越复杂,往往几个 AJAX 请求之间互有依赖,有些请求依赖前面请求的数据,有些请求需要并行进行。还有在类似 Node.js 的后端 JavaScript 环境中,因为需要进行大量 IO 操作,问题更加明显。这个时候使用回调函数来组织代码往往会导致代码难以阅读。
现在比较流行的解决这个问题的方法是使用 Promise,可以将嵌套的回调函数展平。但是写代码和阅读依然有额外的负担。
另外一个方案是使用 ES6 中新增的 generator,因为 generator 的本质是可以将一个函数执行暂停,并保存上下文,再次调用时恢复当时的状态。co 模块是个不错的封装。但是这样略微有些滥用 generator 特性的感觉。
ES7 中有了更加标准的解决方案,新增了 async/await 两个关键词。async 可以声明一个异步函数,此函数需要返回一个 Promise 对象。await可以等待一个 Promise 对象 resolve,并拿到结果。
比如下面的例子,以往我们无法在 JavaScript 中使用常见的 sleep 函数,只能使用 setTimeout 来注册一个回调函数,在指定的时间之后再执行。有了 async/await 之后,我们就可以这样实现了:
async function sleep(timeout) { return new Promise((resolve, reject) => { setTimeout(function() { resolve(); }, timeout); }); } (async function() { console.log('Do some thing, ' + new Date()); await sleep(3000); console.log('Do other things, ' + new Date()); })();
执行此段代码,可以在终端中看到结果:
Do some thing, Mon Feb 23 2015 21:52:11 GMT+0800 (CST)
Do other things, Mon Feb 23 2015 21:52:14 GMT+0800 (CST)
另外 async 函数可以正常的返回结果和抛出异常。await 函数调用即可拿到结果,在外面包上 try/catch 就可以捕获异常。下面是一个从豆瓣 API 获取数据的例子:
var fetchDoubanApi = function() { return new Promise((resolve, reject) => { var xhr = new XMLHttpRequest(); xhr.onreadystatechange = function() { if (xhr.readyState === 4) { if (xhr.status >= 200 && xhr.status < 300) { var response; try { response = JSON.parse(xhr.responseText); } catch (e) { reject(e); } if (response) { resolve(response, xhr.status, xhr); } } else { reject(xhr); } } }; xhr.open('GET', 'https://api.douban.com/v2/user/aisk', true); xhr.setRequestHeader("Content-Type", "text/plain"); xhr.send(data); }); }; (async function() { try { let result = await fetchDoubanApi(); console.log(result); } catch (e) { console.log(e); } })();
async 函数的用法
同 Generator 函数一样,async 函数返回一个 Promise 对象,可以使用 then 方法添加回调函数。当函数执行的时候,一旦遇到 await 就会先返回,等到触发的异步操作完成,再接着执行函数体内后面的语句。
下面是一个例子。
async function getStockPriceByName(name) { var symbol = await getStockSymbol(name); var stockPrice = await getStockPrice(symbol); return stockPrice; } getStockPriceByName('goog').then(function (result){ console.log(result); });
阅读本文前,期待您对promise和ES6(ECMA2015)有所了解,会更容易理解。本文以体验为主,不会深入说明,结尾有详细的文章引用。
第一个例子
Async/Await应该是目前最简单的异步方案了,首先来看个例子。这里我们要实现一个暂停功能,输入N毫秒,则停顿N毫秒后才继续往下执行。
var sleep = function (time) { return new Promise(function (resolve, reject) { setTimeout(function () { resolve(); }, time); }) }; var start = async function () { // 在这里使用起来就像同步代码那样直观 console.log('start'); await sleep(3000); console.log('end'); }; start();
控制台先输出start,稍等3秒后,输出了end。
基本规则
async 表示这是一个async函数,await只能用在这个函数里面。await表示在这里等待promise返回结果了,再继续执行。await 后面跟着的应该是一个promise对象(当然,其他返回值也没关系,只是会立即执行,不过那样就没有意义了…)
获得返回值
await等待的虽然是promise对象,但不必写.then(..),直接可以得到返回值。
var sleep = function (time) { return new Promise(function (resolve, reject) { setTimeout(function () { // 返回 ‘ok' resolve('ok'); }, time); }) }; var start = async function () { let result = await sleep(3000); console.log(result); // 收到 ‘ok' };
捕捉错误
既然.then(..)不用写了,那么.catch(..)也不用写,可以直接用标准的try catch语法捕捉错误。
var sleep = function (time) { return new Promise(function (resolve, reject) { setTimeout(function () { // 模拟出错了,返回 ‘error' reject('error'); }, time); }) }; var start = async function () { try { console.log('start'); await sleep(3000); // 这里得到了一个返回错误 // 所以以下代码不会被执行了 console.log('end'); } catch (err) { console.log(err); // 这里捕捉到错误 `error` } };
循环多个await
await看起来就像是同步代码,所以可以理所当然的写在for循环里,不必担心以往需要闭包才能解决的问题。
..省略以上代码 var start = async function () { for (var i = 1; i <= 10; i++) { console.log(`当前是第${i}次等待..`); await sleep(1000); } };
值得注意的是,await必须在async函数的上下文中的。
..省略以上代码 let one2ten = [1,2,3,4,5,6,7,8,9,10]; // 错误示范 one2ten.forEach(function (v) { console.log(`当前是第${v}次等待..`); await sleep(1000); // 错误!! await只能在async函数中运行 }); // 正确示范 for(var v of one2ten) { console.log(`当前是第${v}次等待..`); await sleep(1000); // 正确, for循环的上下文还在async函数中 }
以上がES7のasync/await使用例を詳しく解説の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。