掌握Node.js中的Async和Await
在本文中,你將學習如何使用Node.js中的async函數(async/await)來簡化callback或Promise.
異步語言結構在其他語言中已經存在了,像c#的async/await、Kotlin的coroutines、go的goroutines,隨著Node.js 8的發布,期待已久的async函數也在其中默認實現了。
Node中的async函數是什么?
當函數聲明為一個Async函數它會返回一個AsyncFunction對象,它們類似于Generator因為執可以被暫停。唯一的區別是它們返回的是Promise而不是{ value: any, done: Boolean }對象。不過它們還是非常相似,你可以使用co包來獲取同樣的功能。
在async函數中,可以等待Promise完成或捕獲它拒絕的原因。
如果你要在Promise中實現一些自己的邏輯的話
function handler (req, res) {return request('https://user-handler-service').catch((err) => {logger.error('Http error', err)error.logged = truethrow err}).then((response) => Mongo.findOne({ user: response.body.user })).catch((err) => {!error.logged && logger.error('Mongo error', err)error.logged = truethrow err}).then((document) => executeLogic(req, res, document)).catch((err) => {!error.logged && console.error(err)res.status(500).send()}) } 復制代碼可以使用async/await讓這個代碼看起來像同步執行的代碼
async function handler (req, res) {let responsetry {response = await request('https://user-handler-service') } catch (err) {logger.error('Http error', err)return res.status(500).send()}let documenttry {document = await Mongo.findOne({ user: response.body.user })} catch (err) {logger.error('Mongo error', err)return res.status(500).send()}executeLogic(document, req, res) } 復制代碼在老的v8版本中,如果有有個promise的拒絕沒有被處理你會得到一個警告,可以不用創建一個拒絕錯誤監聽函數。然而,建議在這種情況下退出你的應用程序。因為當你不處理錯誤時,應用程序處于一個未知的狀態。
process.on('unhandledRejection', (err) => { console.error(err)process.exit(1) }) 復制代碼async函數模式
在處理異步操作時,有很多例子讓他們就像處理同步代碼一樣。如果使用Promise或callbacks來解決問題時需要使用很復雜的模式或者外部庫。
當需要再循環中使用異步獲取數據或使用if-else條件時就是一種很復雜的情況。
指數回退機制
使用Promise實現回退邏輯相當笨拙
function requestWithRetry (url, retryCount) {if (retryCount) {return new Promise((resolve, reject) => {const timeout = Math.pow(2, retryCount)setTimeout(() => {console.log('Waiting', timeout, 'ms')_requestWithRetry(url, retryCount).then(resolve).catch(reject)}, timeout)})} else {return _requestWithRetry(url, 0)} }function _requestWithRetry (url, retryCount) {return request(url, retryCount).catch((err) => {if (err.statusCode && err.statusCode >= 500) {console.log('Retrying', err.message, retryCount)return requestWithRetry(url, ++retryCount)}throw err}) }requestWithRetry('http://localhost:3000').then((res) => {console.log(res)}).catch(err => {console.error(err)}) 復制代碼代碼看的讓人很頭疼,你也不會想看這樣的代碼。我們可以使用async/await重新這個例子,使其更簡單
function wait (timeout) {return new Promise((resolve) => {setTimeout(() => {resolve()}, timeout)}) }async function requestWithRetry (url) {const MAX_RETRIES = 10for (let i = 0; i <= MAX_RETRIES; i++) {try {return await request(url)} catch (err) {const timeout = Math.pow(2, i)console.log('Waiting', timeout, 'ms')await wait(timeout)console.log('Retrying', err.message, i)}} } 復制代碼上面代碼看起來很舒服對不對
中間值
不像前面的例子那么嚇人,如果你有3個異步函數依次相互依賴的情況,那么你必須從幾個難看的解決方案中進行選擇。
functionA返回一個Promise,那么functionB需要這個值而functioinC需要functionA和functionB完成后的值。
方案1:then 圣誕樹
function executeAsyncTask () {return functionA().then((valueA) => {return functionB(valueA).then((valueB) => { return functionC(valueA, valueB)})}) } 復制代碼用這個解決方案,我們在第三個then中可以獲得valueA和valueB,然后可以向前面兩個then一樣獲得valueA和valueB的值。這里不能將圣誕樹(毀掉地獄)拉平,如果這樣做的話會丟失閉包,valueA在functioinC中將不可用。
方案2:移動到上一級作用域
function executeAsyncTask () {let valueAreturn functionA().then((v) => {valueA = vreturn functionB(valueA)}).then((valueB) => {return functionC(valueA, valueB)}) } 復制代碼在這顆圣誕樹中,我們使用更高的作用域保變量valueA,因為valueA作用域在所有的then作用域外面,所以functionC可以拿到第一個functionA完成的值。
這是一個很有效扁平化.then鏈"正確"的語法,然而,這種方法我們需要使用兩個變量valueA和v來保存相同的值。
方案3:使用一個多余的數組
function executeAsyncTask () {return functionA().then(valueA => {return Promise.all([valueA, functionB(valueA)])}).then(([valueA, valueB]) => {return functionC(valueA, valueB)}) } 復制代碼在函數functionA的then中使用一個數組將valueA和Promise一起返回,這樣能有效的扁平化圣誕樹(回調地獄)。
方案4:寫一個幫助函數
const converge = (...promises) => (...args) => {let [head, ...tail] = promisesif (tail.length) {return head(...args).then((value) => converge(...tail)(...args.concat([value])))} else {return head(...args)} }functionA(2).then((valueA) => converge(functionB, functionC)(valueA)) 復制代碼這樣是可行的,寫一個幫助函數來屏蔽上下文變量聲明。但是這樣的代碼非常不利于閱讀,對于不熟悉這些魔法的人就更難了。
使用async/await我們的問題神奇般的消失
async function executeAsyncTask () {const valueA = await functionA()const valueB = await functionB(valueA)return function3(valueA, valueB) } 復制代碼使用async/await處理多個平行請求
和上面一個差不多,如果你想一次執行多個異步任務,然后在不同的地方使用它們的值可以使用async/await輕松搞定。
async function executeParallelAsyncTasks () {const [ valueA, valueB, valueC ] = await Promise.all([ functionA(), functionB(), functionC() ])doSomethingWith(valueA)doSomethingElseWith(valueB)doAnotherThingWith(valueC) } 復制代碼數組迭代方法
你可以在map、filter、reduce方法中使用async函數,雖然它們看起來不是很直觀,但是你可以在控制臺中實驗以下代碼。
1.map
function asyncThing (value) {return new Promise((resolve, reject) => {setTimeout(() => resolve(value), 100)}) }async function main () {return [1,2,3,4].map(async (value) => {const v = await asyncThing(value)return v * 2}) }main().then(v => console.log(v)).catch(err => console.error(err)) 復制代碼2.filter
function asyncThing (value) {return new Promise((resolve, reject) => {setTimeout(() => resolve(value), 100)}) }async function main () {return [1,2,3,4].filter(async (value) => {const v = await asyncThing(value)return v % 2 === 0}) }main().then(v => console.log(v)).catch(err => console.error(err)) 復制代碼3.reduce
function asyncThing (value) {return new Promise((resolve, reject) => {setTimeout(() => resolve(value), 100)}) }async function main () {return [1,2,3,4].reduce(async (acc, value) => {return await acc + await asyncThing(value)}, Promise.resolve(0)) }main().then(v => console.log(v)).catch(err => console.error(err)) 復制代碼解決方案:
[ Promise { <pending> }, Promise { <pending> }, Promise { <pending> }, Promise { <pending> } ]
[ 1, 2, 3, 4 ]
10
如果是map迭代數據你會看到返回值為[ 2, 4, 6, 8 ],唯一的問題是每個值被AsyncFunction函數包裹在了一個Promise中
所以如果想要獲得它們的值,需要將數組傳遞給Promise.All()來解開Promise的包裹。
main().then(v => Promise.all(v)).then(v => console.log(v)).catch(err => console.error(err)) 復制代碼一開始你會等待Promise解決,然后使用map遍歷每個值
function main () {return Promise.all([1,2,3,4].map((value) => asyncThing(value))) }main().then(values => values.map((value) => value * 2)).then(v => console.log(v)).catch(err => console.error(err)) 復制代碼這樣好像更簡單一些?
如果在你的迭代器中如果你有一個長時間運行的同步邏輯和另一個長時間運行的異步任務,async/await版本任然常有用
這種方式當你能拿到第一個值,就可以開始做一些計算,而不必等到所有Promise完成才運行你的計算。盡管結果包裹在Promise中,但是如果按順序執行結果會更快。
關于filter的問題
你可能發覺了,即使上面filter函數里面返回了[ false, true, false, true ],await asyncThing(value)會返回一個promise那么你肯定會得到一個原始的值。你可以在return之前等待所有異步完成,在進行過濾。
Reducing很簡單,有一點需要注意的就是需要將初始值包裹在Promise.resolve中
重寫基于callback的node應用成
Async函數默認返回一個Promise,所以你可以使用Promises來重寫任何基于callback的函數,然后await等待他們執行完畢。在node中也可以使用util.promisify函數將基于回調的函數轉換為基于Promise的函數
重寫基于Promise的應用程序
要轉換很簡單,.then將Promise執行流串了起來。現在你可以直接使用`async/await。
function asyncTask () {return functionA().then((valueA) => functionB(valueA)).then((valueB) => functionC(valueB)).then((valueC) => functionD(valueC)).catch((err) => logger.error(err)) }復制代碼轉換后
async function asyncTask () {try {const valueA = await functionA()const valueB = await functionB(valueA)const valueC = await functionC(valueB)return await functionD(valueC)} catch (err) {logger.error(err)} } Rewriting Nod 復制代碼使用Async/Await將很大程度上的使應用程序具有高可讀性,降低應用程序的處理復雜度(如:錯誤捕獲),如果你也使用 node v8+的版本不妨嘗試一下,或許會有新的收獲。
如有錯誤麻煩留言告訴我進行改正,謝謝閱讀
原文鏈接
總結
以上是生活随笔為你收集整理的掌握Node.js中的Async和Await的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Linux性能查看
- 下一篇: 通过ssh建立点对点的隧道,实现两个子网