欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

一文詳解如何有效的處理Promise并發(fā)

 更新時間:2023年04月19日 10:17:57   作者:liangyue  
這篇文章主要為大家介紹如何有效的處理Promise并發(fā)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪

前言

如上圖所示的代碼,相信大家在平時的開發(fā)中肯定用到不少,我們可以進行優(yōu)化,使得只用短短一半的時間就可以完成。 在上面的函數(shù)中,我們等待用戶信息請求完成后再請求詳情信息。但這兩個函數(shù)之間并沒有任何關(guān)聯(lián),所以我們可以同時觸發(fā)兩個請求,并同時等待完成。那么,我們有多少種方式來實現(xiàn)呢?你真的可以處理好Promise的并發(fā)么?接下來,我們進入正題。

Promise.all

如何實現(xiàn)

Promise.all 大家應該是比較熟悉的,Promise.all方法接收一個promiseiterable類型,如果所有傳入的promise都變成完成狀態(tài),Promise.all返回的Promise異步的變?yōu)橥瓿?。如果傳入?code>promise中有一個失敗(rejected),Promise.all將失敗的結(jié)果給失敗狀態(tài)的回調(diào)函數(shù),而不管其他promise是否完成。 我們可以如下實現(xiàn)剛才的代碼

async function init() {
  const [user, info] = await Promise.all([
    getUser(),
    getInfo()
  ])
  console.log('init', user, info)
}

現(xiàn)在這種方式,如果我們之前每個請求都需要1秒,一共需要2秒,那么現(xiàn)在兩個同時執(zhí)行只需要1秒就完成了!但是這樣也有一個問題:我們并沒有考慮報錯問題。你可能會認為,這個很簡單,把代碼放在一個try...cahtch中不就可以了,就像這樣:

async function init() {
  try {
    const [user, info] = await Promise.all([
      getUser(),
      getInfo()
    ])
    console.log('init ==== ', user, info)
  } catch(err) {
    console.log('err', err);
  }
}

但是,這樣的話會有一個問題,就像這樣:

function getUser() {
  return new Promise((resolve, reject) => {
    setTimeout(() => {
      reject('user reject')
    }, 500);
  })
}
function getInfo() {
  return new Promise((resolve, reject) => {
    setTimeout(() => {
      reject('info reject')
    }, 1000);
  })
}
// 輸出 err user reject

由于getUser優(yōu)先完成并出現(xiàn)錯誤,此時觸發(fā)了catch,而當getInfo再次完成并出現(xiàn)錯誤時,將不會觸發(fā)catch。因為catch代碼已經(jīng)運行,函數(shù)已經(jīng)完成。 那么,要怎么做呢?接下來,我們就講講應該如何處理報錯問題。

如何處理報錯

解決方式是,給Promise.all中的每個函數(shù)加上catch,如下:

function handle(err) {
  console.log('err', err)
}
function onReject(err) {
  handle(err);
  return new Error(err);
}
async function init() {
  const [user, info] = await Promise.all([
    getUser().catch(onReject),
    getInfo().catch(onReject)
  ])
  console.log('init', user instanceof Error, info instanceof Error) // init true true
}

這樣,我們在onReject函數(shù)中處理錯誤,并返回這個錯誤。所以現(xiàn)在我們生成的userinfo要么是Error要么是我們期望的效果,而Error我們可以用instanceof檢查它。

Promise.allSettled

解決并發(fā)我們還可以使用 Promise.allSettled ,我們會得到一個包含每個Promise結(jié)果的值或錯誤信息。

如何實現(xiàn)

接下來,我們來使用一下,代碼如下:

async function init() {
  const [userStatus, infoStatus] = await Promise.allSettled([
    getUser(),
    getInfo()
  ])
  console.log('info', userStatus, infoStatus)
}

現(xiàn)在,我們可以得到這樣的數(shù)據(jù):

結(jié)果對象有3個屬性:

  • status: fulfilledrejected
  • value: 僅在statusfulfilled時出現(xiàn),為Promiseresolve返回的值
  • reason: 僅在statusrejected時出現(xiàn),為Promisereject時返回的值

因此,我們可以讀取到每個Promise的狀態(tài),并單獨處理每個錯誤而不會遺漏任何的信息。

最后兩個技巧

Promise.race

Promise.race方法可接受一個可迭代的promise返回一個promise,一旦迭代器中某個promise解決或拒絕,返回的promise就會resolvereject。

我們可以這樣實現(xiàn)一個簡單的超時功能,代碼如下:

function getUser() {
  return new Promise((resolve) => {
    setTimeout(() => {
      resolve('user resolve')
    }, 5100);
  })
}
async function init() {
  // Race to see which Promise completes first
  const racePromise = Promise.race([
    getUser(),
    new Promise((resolve, reject) =>
      // Time out after 5 seconds
      setTimeout(() => reject(new Error('Timeout')), 5000)
    )
  ])
  try {
    const result = await racePromise
    console.log('result', result)
  } catch (err) {
    console.log('err', err)
    // Timed out!
  }
}

注意,通常情況下,如果有超時,那么你需要盡量的取消未完成的待處理任務(wù)。

另外,最好還是處理所有promisereject

const racePromise = Promise.race([
    getUser().catch(onReject),
    // xxx
  ])

Promise.any

Promise.any 等待任何一個promise成功則為成功,只有全部的promise都被reject,才會返回reject。通常我們可以使用Promise.any來實現(xiàn),當一個promise先完成后,取消其他的promise,不過要注意的是,我們并不總是同時要處理多個數(shù)據(jù),只是因為我們可以做到,所以要謹慎的使用它。

通常,我們不想出現(xiàn)未被處理的reject,所以,我們應該這樣寫:

const anyPromise = Promise.any([
    getUser().catch(onReject),
    getInfo().catch(onReject)
])

總結(jié)

前面我們已經(jīng)介紹了在使用promise處理并發(fā)問題時,應該如何的處理promisereject。但是也需要注意:過度的并發(fā)會導致網(wǎng)絡(luò)抖動、磁盤抖動或其他問題,雖然可以,但并不是一定要這樣做,有時使用async await也可以使代碼更容易理解和維護,所以,在使用之前我們更需要考慮是否需要。

以上就是一文詳解如何有效的處理Promise并發(fā)的詳細內(nèi)容,更多關(guān)于有效處理Promise并發(fā)的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Bootstrap3制作圖片輪播效果

    Bootstrap3制作圖片輪播效果

    這篇文章主要教大家如何利用Bootstrap3制作圖片輪播效果,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2016-05-05
  • 基于Echarts實現(xiàn)繪制立體柱狀圖的示例代碼

    基于Echarts實現(xiàn)繪制立體柱狀圖的示例代碼

    這篇文章主要為大家詳細介紹了如何基于Echarts實現(xiàn)繪制立體柱狀圖的功能,文中的示例代碼講解詳細,具有一定的借鑒價值,需要的可以參考一下
    2023-02-02
  • JavaScript中的普通函數(shù)和箭頭函數(shù)的區(qū)別和用法詳解

    JavaScript中的普通函數(shù)和箭頭函數(shù)的區(qū)別和用法詳解

    這篇文章主要介紹了JavaScript中的普通函數(shù)和箭頭函數(shù)的區(qū)別和用法詳解,非常不錯,具有參考借鑒價值,需要的朋友可以參考下
    2017-03-03
  • 利用webpack理解CommonJS和ES Modules的差異區(qū)別

    利用webpack理解CommonJS和ES Modules的差異區(qū)別

    這篇文章主要介紹了利用webpack理解CommonJS和ES Modules的差異區(qū)別,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2020-06-06
  • JavaScript獲取本機IP地址三種常用方式

    JavaScript獲取本機IP地址三種常用方式

    在瀏覽器中使用JavaScript獲取本機IP地址是不可行的,因為瀏覽器的安全策略禁止JavaScript訪問這樣的信息,這篇文章主要給大家介紹了關(guān)于JavaScript獲取本機IP地址的三種常用方式,需要的朋友可以參考下
    2023-12-12
  • JavaScript中代理與反射的用法詳解

    JavaScript中代理與反射的用法詳解

    JavaScript作為一門靈活而強大的語言,提供了代理(Proxy)與反射(Reflect)這兩個元編程工具,它們?yōu)殚_發(fā)者提供了更深層次的語言控制和操作,在本篇博客中,我們將深入研究代理與反射的概念、用法,以及如何巧妙地結(jié)合它們來實現(xiàn)高級的編程技巧,需要的朋友可以參考下
    2023-12-12
  • JavaScript中setTimeout()的具體用法

    JavaScript中setTimeout()的具體用法

    本文主要介紹了JavaScript中setTimeout()的具體用法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2023-04-04
  • 對于js垃圾回收機制的理解

    對于js垃圾回收機制的理解

    下面小編就為大家?guī)硪黄獙τ趈s垃圾回收機制的理解。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-09-09
  • 仿YAHOO的一個小效果 下拉效果

    仿YAHOO的一個小效果 下拉效果

    [紅色]仿YAHOO的一個小效果 下拉效果...
    2006-11-11
  • JavaScript基于對象方法實現(xiàn)數(shù)組去重及排序操作示例

    JavaScript基于對象方法實現(xiàn)數(shù)組去重及排序操作示例

    這篇文章主要介紹了JavaScript基于對象方法實現(xiàn)數(shù)組去重及排序操作,涉及javascript基于對象方法的數(shù)組遍歷、比較、去重、排序等相關(guān)操作技巧,需要的朋友可以參考下
    2018-07-07

最新評論