真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

async/await地獄該如何避免詳解

前言

成都創(chuàng)新互聯(lián)專注為客戶提供全方位的互聯(lián)網(wǎng)綜合服務(wù),包含不限于成都網(wǎng)站制作、網(wǎng)站設(shè)計(jì)、三元網(wǎng)絡(luò)推廣、成都微信小程序、三元網(wǎng)絡(luò)營銷、三元企業(yè)策劃、三元品牌公關(guān)、搜索引擎seo、人物專訪、企業(yè)宣傳片、企業(yè)代運(yùn)營等,從售前售中售后,我們都將竭誠為您服務(wù),您的肯定,是我們最大的嘉獎(jiǎng);成都創(chuàng)新互聯(lián)為所有大學(xué)生創(chuàng)業(yè)者提供三元建站搭建服務(wù),24小時(shí)服務(wù)熱線:028-86922220,官方網(wǎng)址:www.cdcxhl.com

async/await是什么

async/await可以說是co模塊和生成器函數(shù)的語法糖。用更加清晰的語義解決js異步代碼。

熟悉co模塊的同學(xué)應(yīng)該都知道,co模塊是TJ大神寫的一個(gè)使用生成器函數(shù)來解決異步流程的模塊,可以看做是生成器函數(shù)的執(zhí)行器。而async/await則是對(duì)co模塊的升級(jí),內(nèi)置生成器函數(shù)的執(zhí)行器,不再依賴co模塊。同時(shí),async返回的是Promise。

從上面來看,不管是co模塊還是async/await,都是將Promise作為最基礎(chǔ)的單元,對(duì)Promise不很了解的同學(xué)可以先深入了解一下Promise。

async/await 寫著很爽,不過要注意這些問題。

async/await 讓我們擺脫了回調(diào)地獄,但是這又引入了 async/await 地獄的問題。

async/await地獄該如何避免詳解

什么是 async/await 地獄

在 Javascript 中進(jìn)行異步編程的時(shí)候,人們總是使用很多 await 語句,很多時(shí)候我們的語句并不需要依賴于之前的語句,這樣就會(huì)導(dǎo)致性能問題。

async/await 地獄的例子

我們?cè)囍鴮懸粋€(gè)購買披薩和飲料的程序:

(async () => {
 const pizzaData = await getPizzaData() // async call
 const drinkData = await getDrinkData() // async call
 const chosenPizza = choosePizza() // sync call
 const chosenDrink = chooseDrink() // sync call
 await addPizzaToCart(chosenPizza) // async call
 await addDrinkToCart(chosenDrink) // async call
 orderItems() // async call
})()

這段代碼運(yùn)行沒有問題。但是不是一個(gè)好的實(shí)現(xiàn),因?yàn)檫@增加了不必要的等待。

說明

我們已經(jīng)將我們的代碼封裝在異步 IIFE 中,按照下面的順序執(zhí)行:

得到披薩名單
獲取飲料列表
從列表中選擇一個(gè)披薩
從列表中選擇一種飲料
將選中的披薩加入購物車
將選擇的飲品加入購物車
訂購購物車中的物品

問題

這里有個(gè)問題為什么從列表中選擇披薩這個(gè)動(dòng)作要等待獲取飲料列表?這兩個(gè)是沒什么關(guān)聯(lián)的操作。其中的關(guān)聯(lián)操作有兩組:

獲取披薩列表 -》 選擇披薩 -》 選擇披薩加入購物車

獲取飲料列表 -》 選擇飲料 -》 選擇飲料加入購物車

這兩組操作應(yīng)該是并發(fā)執(zhí)行的。

再來看一個(gè)更差的例子

這個(gè) Javascript 代碼片段將購物車中的商品并發(fā)出訂購請(qǐng)求。

async function orderItems() {
 const items = await getCartItems() // async call
 const noOfItems = items.length
 for(var i = 0; i < noOfItems; i++) {
 await sendRequest(items[i]) // async call
 }
}

這種情況 for 循環(huán)必須等待 sendRequest() 函數(shù)完成才能繼續(xù)下一次迭代。但是,我們并不需要等待。我們希望盡快發(fā)送所有請(qǐng)求。然后我們可以等待所有請(qǐng)求完成。

現(xiàn)在你應(yīng)該已經(jīng)對(duì) async/await 地獄有跟多的了解,現(xiàn)在我們?cè)賮砜紤]一個(gè)問題

如果我們忘記 await 關(guān)鍵字會(huì)怎么樣?

如果在調(diào)用異步函數(shù)忘記使用 await,這意味著執(zhí)行該功能不需要等待。異步函數(shù)將直接返回一個(gè) promise,你可以稍后使用。

(async () => {
 const value = doSomeAsyncTask()
 console.log(value) // an unresolved promise
})()

或者是程序不清楚你想要等待函數(shù)執(zhí)行完,直接退出不會(huì)完成這個(gè)異步任務(wù)。所以我們需要使用 await 這個(gè)關(guān)鍵字。

promise 有一個(gè)有趣的屬性,你可以在某行代碼中獲取 promise,然后在其他地方中等待它 resolve,這是解決 async/await 地獄的關(guān)鍵。

(async () => {
 const promise = doSomeAsyncTask()
 const value = await promise
 console.log(value) // the actual value
})()

如你所見 doSomeAsyncTask 直接返回一個(gè) Promise 同時(shí)這個(gè)異步函數(shù) doSomeAsyncTask 已經(jīng)開始執(zhí)行,為了得到 doSomeAsyncTask 的返回值,我們需要 await 來告訴

應(yīng)該如何避免 async/await 地獄

首先我們需要知道哪些命名是有前后依賴關(guān)系的。

然后將有依賴關(guān)系的系列操作進(jìn)行分組合并成一個(gè)異步操作。

同時(shí)執(zhí)行這些異步函數(shù)。

我們來重寫這寫例子:

async function selectPizza() {
 const pizzaData = await getPizzaData() // async call
 const chosenPizza = choosePizza() // sync call
 await addPizzaToCart(chosenPizza) // async call
}

async function selectDrink() {
 const drinkData = await getDrinkData() // async call
 const chosenDrink = chooseDrink() // sync call
 await addDrinkToCart(chosenDrink) // async call
}

(async () => {
 const pizzaPromise = selectPizza()
 const drinkPromise = selectDrink()
 await pizzaPromise
 await drinkPromise
 orderItems() // async call
})()

// Although I prefer it this way

(async () => {
 Promise.all([selectPizza(), selectDrink()].then(orderItems) // async call
})()

我們將語句分成兩個(gè)函數(shù)。在函數(shù)內(nèi)部,每個(gè)語句都依賴于前一個(gè)語句的執(zhí)行。然后我們同時(shí)執(zhí)行這兩個(gè)函數(shù) selectPizza()和selectDrink() 。

在第二個(gè)例子中我們需要處理未知數(shù)量的 Promise。處理這個(gè)問題非常簡單,我們只需要?jiǎng)?chuàng)建一個(gè)數(shù)組將所有 Promise 存入其中,使用 Promise.all() 方法并行執(zhí)行:

async function orderItems() {
 const items = await getCartItems() // async call
 const noOfItems = items.length
 const promises = []
 for(var i = 0; i < noOfItems; i++) {
 const orderPromise = sendRequest(items[i]) // async call
 promises.push(orderPromise) // sync call
 }
 await Promise.all(promises) // async call
}

總結(jié)

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流,謝謝大家對(duì)創(chuàng)新互聯(lián)的支持。


分享名稱:async/await地獄該如何避免詳解
URL地址:http://weahome.cn/article/iphisd.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部