Promise簡介锋边、promise.all與race應(yīng)用

本人前端小白,項(xiàng)目經(jīng)歷較少旋圆,遇到頁面接口請求較多時(shí)總會有些疑惑宠默,不知道怎么處理合理×榍桑看之前的項(xiàng)目代碼,寫了不知多少回調(diào)抹沪,最近整理promise.all和async/await兩個(gè)方法刻肄,分享一下。同時(shí)結(jié)合兩者優(yōu)化下接口性能融欧,減少請求耗時(shí)敏弃。

Promise

眾所周知,Promise 是一個(gè)對象噪馏,用于表示一個(gè)異步操作的最終完成 (或失敗), 及其結(jié)果值麦到。

語法:new Promise( (resolve, reject) => {...} /* executor */ );

let promise = new Promise((resolve,reject) => {
    resolve('success')
     // or
    reject('error')
})

該對象接收兩個(gè)參數(shù)

  • resolve: 將Promise對象的狀態(tài)從“未完成”變?yōu)椤俺晒Α保磸?pending 變?yōu)?resolved)绿饵,在異步操作成功時(shí)調(diào)用,并將異步操作的結(jié)果瓶颠,作為參數(shù)傳遞出去.
  • reject: 將Promise對象的狀態(tài)從“未完成”變?yōu)椤笆 保磸?pending 變?yōu)?rejected)拟赊,在異步操作失敗時(shí)調(diào)用,并將異步操作報(bào)出的錯(cuò)誤粹淋,作為參數(shù)傳遞出去吸祟。
promise.then(res => {
    console.log(res)
}).catch(err => {
    console.log(err)
})

then方法會返回一個(gè)Promise,接收一個(gè)回調(diào)函數(shù)作參數(shù),回調(diào)函數(shù)接受Promise對象傳出的值作為參數(shù)桃移。

Promise.all()

將多個(gè)promise實(shí)例包裝成一個(gè)新的promise實(shí)例屋匕,接收一個(gè)數(shù)組作參數(shù)。成功時(shí)返回一個(gè)數(shù)組借杰,數(shù)組中包括所有子promise的結(jié)果过吻;失敗時(shí),返回第一個(gè)失敗promise的結(jié)果蔗衡。

let p1 = new Promise((resolve,reject) => {
   resolve('success1')
   //reject('error')
})
let p2 = new Promise((resolve,reject) => {
    resolve('success2')
 })
 let p3 = new Promise((resolve,reject) => {
    reject('error')
 })

成功狀態(tài)時(shí):

Promise.all([p1,p2]).then(res => {
     console.log(res)
 }).catch(err => {
     console.log(err)
 })
 
 // 返回結(jié)果
 
 ['success1', 'success2' ]

失敗時(shí):

 Promise.all([p1,p2,p3]).then(res => {
     console.log(res)
 }).catch(err => {
     console.log(err)
 })
 
 // 輸出
 error

注意:

  • 成功時(shí)返回?cái)?shù)組中疮装,值的順序跟傳入?yún)?shù)中promise實(shí)例的順序?qū)?yīng),類似于同步執(zhí)行粘都,不會因?yàn)榉祷亟Y(jié)果耗時(shí)的影響廓推。

適用場景:

  • 頁面請求過多時(shí),可根據(jù)Promise.all()返回結(jié)果判斷所有請求是否成功翩隧,否則使頁面一直處于loading狀態(tài)樊展,避免用戶操作,或者提是錯(cuò)誤堆生。
  • 發(fā)送多個(gè)請求专缠,需要根據(jù)請求順序獲取和使用數(shù)據(jù)的場景。
  • 避免promise使用不合理或各種回調(diào)函數(shù)造成地獄回調(diào)淑仆,無法維護(hù)涝婉。

Promise.race()

用法同Promise.all(),哪個(gè)子promise結(jié)果先被返回,就返回該結(jié)果蔗怠,不管是成功還是失敗狀態(tài)墩弯。

let p1 = new Promise((resolve, reject) => {
  setTimeout(() => {
    resolve('success')
  },1000)
})

let p2 = new Promise((resolve, reject) => {
  setTimeout(() => {
    reject('error')
  }, 500)
})

Promise.race([p1, p2]).then((result) => {
  console.log(result)
}).catch((error) => {
  console.log(error)  // 'error'
})

async/await

  • 見名知意: 異步與等待
  • 同步思維解決異步問題的方案。

async

函數(shù)使用async后將轉(zhuǎn)換成Promise寞射,并且其返回值也是一個(gè)Promise渔工。

async function fn() {
    console.log('async')
    return 1
}
fn()

// 結(jié)果
async
Promise {<resolved>: 1}

如果函數(shù)內(nèi)部未使用return,resolved的值為undefind。(最好在瀏覽器上嘗試桥温,node上有返回promise引矩,但無法顯示,可能我的node4版本較低)

  • 既然返回值是Promise,那我們變可以使用then()獲取旺韭。
async function fn() {
    console.log('async')
    return 1
}
fn().then(res => {
    console.log(res) // 1
})

await

  • await只能在async函數(shù)內(nèi)部使用,用在普通函數(shù)里就會報(bào)錯(cuò)

結(jié)合實(shí)際開發(fā)來說明

fetch('https://blog.csdn.net/')
  .then(res => {
    console.log(res)
    return fetch('https://juejin.im/')
  })
  .then(res => {
    console.log(res)
  })
  .catch(error => {
    console.log(error)
  })
  • 這種代碼可能大家都有見過氛谜,雖然能體現(xiàn)等待的效果,但當(dāng)鏈?zhǔn)秸{(diào)用過多区端,將充斥著多個(gè)then,配合業(yè)務(wù)邏輯那就部那么語義化值漫,且難以維護(hù)。
async function foo () {
  try {
    let response1 = await fetch('https://blog.csdn.net/')
    console.log(response1)
    let response2 = await fetch('https://juejin.im/')
    console.log(response2)
  } catch (err) {
    console.error(err)
  }
}
foo()

  • 上面代碼,整個(gè)異步處理的邏輯都是使用同步代碼的方式來實(shí)現(xiàn)的,感覺就在寫同步代碼珊燎,比較符合人的線性思維的惭嚣。
  • await 后面一定是Promise 對象,如果不是會自動(dòng)包裝成Promise對象悔政。

promise.all與async/await結(jié)合晚吞,優(yōu)化請求

先看個(gè)栗子

function fn1() {
    return new Promise(resolve => {
        setTimeout(() => {
            resolve('p1')
        },1000)
    })
}
function fn2() {
    return new Promise(resolve => {
        setTimeout(() => {
            resolve('p2')
        },2000)
    })
}
async function fn() {
    let p1 = await fn1()
    let p2 = await fn2()
}
 
 fn()

忽略瀏覽器引擎的耗時(shí),fn1谋国,fn2理想總共耗時(shí)3000毫秒

function fn1() {
    return new Promise(resolve => {
        setTimeout(() => {
            resolve('p1')
        },1000)
    })
}
function fn2() {
    return new Promise(resolve => {
        setTimeout(() => {
            resolve('p2')
        },2000)
    })
}

async function fn() {
    let [res1, res2] = await Promise.all([test1(), test2()])
}
fn()

而這種寫發(fā)理想耗時(shí)2000ms,因?yàn)閜romise.all是將多個(gè)異步操作并行實(shí)現(xiàn)槽地。
需要注意的是,多個(gè)await命令不存在依賴關(guān)系芦瘾。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末捌蚊,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子近弟,更是在濱河造成了極大的恐慌缅糟,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,858評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件祷愉,死亡現(xiàn)場離奇詭異窗宦,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)二鳄,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,372評論 3 395
  • 文/潘曉璐 我一進(jìn)店門赴涵,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人订讼,你說我怎么就攤上這事髓窜。” “怎么了欺殿?”我有些...
    開封第一講書人閱讀 165,282評論 0 356
  • 文/不壞的土叔 我叫張陵寄纵,是天一觀的道長。 經(jīng)常有香客問我祈餐,道長擂啥,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,842評論 1 295
  • 正文 為了忘掉前任帆阳,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘蜒谤。我一直安慰自己山宾,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,857評論 6 392
  • 文/花漫 我一把揭開白布鳍徽。 她就那樣靜靜地躺著资锰,像睡著了一般。 火紅的嫁衣襯著肌膚如雪阶祭。 梳的紋絲不亂的頭發(fā)上绷杜,一...
    開封第一講書人閱讀 51,679評論 1 305
  • 那天,我揣著相機(jī)與錄音濒募,去河邊找鬼鞭盟。 笑死,一個(gè)胖子當(dāng)著我的面吹牛瑰剃,可吹牛的內(nèi)容都是我干的齿诉。 我是一名探鬼主播,決...
    沈念sama閱讀 40,406評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼晌姚,長吁一口氣:“原來是場噩夢啊……” “哼粤剧!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起挥唠,我...
    開封第一講書人閱讀 39,311評論 0 276
  • 序言:老撾萬榮一對情侶失蹤抵恋,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后宝磨,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體弧关,經(jīng)...
    沈念sama閱讀 45,767評論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,945評論 3 336
  • 正文 我和宋清朗相戀三年懊烤,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了梯醒。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,090評論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡腌紧,死狀恐怖茸习,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情壁肋,我是刑警寧澤号胚,帶...
    沈念sama閱讀 35,785評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站浸遗,受9級特大地震影響猫胁,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜跛锌,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,420評論 3 331
  • 文/蒙蒙 一弃秆、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦菠赚、人聲如沸脑豹。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,988評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽瘩欺。三九已至,卻和暖如春拌牲,著一層夾襖步出監(jiān)牢的瞬間俱饿,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,101評論 1 271
  • 我被黑心中介騙來泰國打工塌忽, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留拍埠,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,298評論 3 372
  • 正文 我出身青樓砚婆,卻偏偏與公主長得像械拍,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子装盯,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,033評論 2 355

推薦閱讀更多精彩內(nèi)容

  • 透徹掌握Promise的使用坷虑,讀這篇就夠了 Promise的重要性我認(rèn)為我沒有必要多講,概括起來說就是必須得掌握埂奈,...
    穿牛仔褲的蚊子閱讀 2,137評論 0 16
  • async 函數(shù) 含義 ES2017 標(biāo)準(zhǔn)引入了 async 函數(shù)迄损,使得異步操作變得更加方便。 async函數(shù)對 ...
    Xyaleo閱讀 1,094評論 0 4
  • 含義 async函數(shù)是Generator函數(shù)的語法糖账磺,它使得異步操作變得更加方便芹敌。 寫成async函數(shù),就是下面這...
    oWSQo閱讀 1,993評論 0 2
  • async 函數(shù) 含義 ES2017 標(biāo)準(zhǔn)引入了 async 函數(shù)垮抗,使得異步操作變得更加方便氏捞。 async 函數(shù)是...
    huilegezai閱讀 1,259評論 0 6
  • async 函數(shù) ES2017 標(biāo)準(zhǔn)引入了 async 函數(shù),使得異步操作變得更加方便冒版。async 函數(shù)是什么液茎?一...
    _羊羽_閱讀 2,108評論 0 1