在本文中,你將學習如何使用Node.js中的async函數(async/await)來簡化callback或Promise.javascript
異步語言結構在其餘語言中已經存在了,像c#的async/await、Kotlin的coroutines、go的goroutines,隨着Node.js 8的發佈,期待已久的async函數也在其中默認實現了。java
當函數聲明爲一個Async函數它會返回一個AsyncFunction
對象,它們相似於Generator
由於執能夠被暫停。惟一的區別是它們返回的是Promise
而不是{ value: any, done: Boolean }
對象。不過它們仍是很是類似,你可使用co包來獲取一樣的功能。node
在async函數中,能夠等待Promise
完成或捕獲它拒絕的緣由。git
若是你要在Promise中實現一些本身的邏輯的話github
function handler (req, res) {
return request('https://user-handler-service')
.catch((err) => {
logger.error('Http error', err)
error.logged = true
throw err
})
.then((response) => Mongo.findOne({ user: response.body.user }))
.catch((err) => {
!error.logged && logger.error('Mongo error', err)
error.logged = true
throw err
})
.then((document) => executeLogic(req, res, document))
.catch((err) => {
!error.logged && console.error(err)
res.status(500).send()
})
}
複製代碼
可使用async/await
讓這個代碼看起來像同步執行的代碼c#
async function handler (req, res) {
let response
try {
response = await request('https://user-handler-service')
} catch (err) {
logger.error('Http error', err)
return res.status(500).send()
}
let document
try {
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)
})
複製代碼
在處理異步操做時,有不少例子讓他們就像處理同步代碼同樣。若是使用Promise
或callbacks
來解決問題時須要使用很複雜的模式或者外部庫。promise
當須要再循環中使用異步獲取數據或使用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 = 10
for (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
完成後的值。
then
聖誕樹function executeAsyncTask () {
return functionA()
.then((valueA) => {
return functionB(valueA)
.then((valueB) => {
return functionC(valueA, valueB)
})
})
}
複製代碼
用這個解決方案,咱們在第三個then
中能夠得到valueA
和valueB
,而後能夠向前面兩個then
同樣得到valueA
和valueB
的值。這裏不能將聖誕樹(毀掉地獄)拉平,若是這樣作的話會丟失閉包,valueA
在functioinC
中將不可用。
function executeAsyncTask () {
let valueA
return functionA()
.then((v) => {
valueA = v
return functionB(valueA)
})
.then((valueB) => {
return functionC(valueA, valueB)
})
}
複製代碼
在這顆聖誕樹中,咱們使用更高的做用域保變量valueA
,由於valueA
做用域在全部的then
做用域外面,因此functionC
能夠拿到第一個functionA
完成的值。
這是一個頗有效扁平化.then
鏈"正確"的語法,然而,這種方法咱們須要使用兩個變量valueA
和v
來保存相同的值。
function executeAsyncTask () {
return functionA()
.then(valueA => {
return Promise.all([valueA, functionB(valueA)])
})
.then(([valueA, valueB]) => {
return functionC(valueA, valueB)
})
}
複製代碼
在函數functionA
的then
中使用一個數組將valueA
和Promise
一塊兒返回,這樣能有效的扁平化聖誕樹(回調地獄)。
const converge = (...promises) => (...args) => {
let [head, ...tail] = promises
if (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函數,雖然它們看起來不是很直觀,可是你能夠在控制檯中實驗如下代碼。
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))
複製代碼
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))
複製代碼
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
中
Async
函數默認返回一個Promise
,因此你可使用Promises
來重寫任何基於callback
的函數,而後await
等待他們執行完畢。在node中也可使用util.promisify
函數將基於回調的函數轉換爲基於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+的版本不妨嘗試一下,或許會有新的收穫。
若有錯誤麻煩留言告訴我進行改正,謝謝閱讀