async-await

一酗宋、async-await和Promise的關(guān)系

async-awaitpromisegenerator的語法糖。只是為了讓我們書寫代碼時(shí)更加流暢品姓,當(dāng)然也增強(qiáng)了代碼的可讀性寝并。簡單來說:async-await 是建立在 promise機(jī)制之上的,并不能取代其地位腹备。

二衬潦、基本語法

async function basicDemo() {
    let result = await Math.random();
    console.log(result);
}

basicDemo();
// 0.6484863241051226
//Promise {[[PromiseStatus]]: "resolved", [[PromiseValue]]: undefined}

上述代碼就是async-await的基本使用形式。有兩個(gè)陌生的關(guān)鍵字async植酥、await镀岛,同時(shí)函數(shù)執(zhí)行結(jié)果似乎返回了一個(gè)promise對象。

async

async用來表示函數(shù)是異步的,定義的函數(shù)會返回一個(gè)promise對象哎媚,可以使用then方法添加回調(diào)函數(shù)。

async function demo01() {
    return 123;
}

demo01().then(val => {
    console.log(val);// 123
});
若 async 定義的函數(shù)有返回值喊儡,return 123;
相當(dāng)于Promise.resolve(123),沒有聲明式的 return則相當(dāng)于執(zhí)行了Promise.resolve();

await

await 可以理解為是 async wait 的簡寫拨与。await 必須出現(xiàn)在 async 函數(shù)內(nèi)部,不能單獨(dú)使用艾猜。

function notAsyncFunc() {
    await Math.random();
}
notAsyncFunc();//Uncaught SyntaxError: Unexpected identifier

await 后面可以跟任何的JS 表達(dá)式买喧。雖然說 await 可以等很多類型的東西,但是它最主要的意圖是用來等待 Promise 對象的狀態(tài)被 resolved匆赃。如果await的是 promise對象會造成異步函數(shù)停止執(zhí)行并且等待 promise 的解決,如果等的是正常的表達(dá)式則立即執(zhí)行淤毛。

function sleep(second) {
    return new Promise((resolve, reject) => {
        setTimeout(() => {
            resolve(' enough sleep~');
        }, second);
    })
}
function normalFunc() {
    console.log('normalFunc');
}
async function awaitDemo() {
    await normalFunc();
    console.log('something, ~~');
    let result = await sleep(2000);
    console.log(result);// 兩秒之后會被打印出來
}
awaitDemo();
// normalFunc
// VM4036:13 something, ~~
// VM4036:15  enough sleep~

二、實(shí)例

舉例說明啊算柳,你有三個(gè)請求需要發(fā)生低淡,第三個(gè)請求是依賴于第二個(gè)請求的解構(gòu)第二個(gè)請求依賴于第一個(gè)請求的結(jié)果。若用 ES5實(shí)現(xiàn)會有3層的回調(diào)瞬项,若用Promise 實(shí)現(xiàn)至少需要3個(gè)then蔗蹋。一個(gè)是代碼橫向發(fā)展,另一個(gè)是縱向發(fā)展囱淋。今天指給出 async-await 的實(shí)現(xiàn)哈~

//我們?nèi)匀皇褂?setTimeout 來模擬異步請求
function sleep(second, param) {
   return new Promise((resolve, reject) => {
       setTimeout(() => {
           resolve(param);
       }, second);
   })
}

async function test() {
   let result1 = await sleep(2000, 'req01');
   let result2 = await sleep(1000, 'req02' + result1);
   let result3 = await sleep(500, 'req03' + result2);
   console.log(`
       ${result3}
       ${result2}
       ${result1}
   `);
}

test();
//req03req02req01
//req02req01
//req01

三猪杭、錯(cuò)誤處理

上述的代碼好像給的都是resolve的情況,那么reject的時(shí)候我們該如何處理呢妥衣?

function sleep(second) {
    return new Promise((resolve, reject) => {
        setTimeout(() => {
            reject('want to sleep~');
        }, second);
    })
}

async function errorDemo() {
    let result = await sleep(1000);
    console.log(result);
}
errorDemo();// VM706:11 Uncaught (in promise) want to sleep~

// 為了處理Promise.reject 的情況我們應(yīng)該將代碼塊用 try catch 包裹一下
async function errorDemoSuper() {
    try {
        let result = await sleep(1000);
        console.log(result);
    } catch (err) {
        console.log(err);
    }
}

errorDemoSuper();// want to sleep~
// 有了 try catch 之后我們就能夠拿到 Promise.reject 回來的數(shù)據(jù)了皂吮。

四、小心你的并行處理!

對于初學(xué)者來說一不小心就將 ajax 的并發(fā)請求發(fā)成了阻塞式同步的操作了税手,我就真真切切的在工作中寫了這樣的代碼蜂筹。await 若等待的是 promise 就會停止下來。業(yè)務(wù)是這樣的芦倒,我有三個(gè)異步請求需要發(fā)送狂票,相互沒有關(guān)聯(lián),只是需要當(dāng)請求都結(jié)束后將界面的 loading 清除掉即可熙暴。
剛學(xué)完 async await 開心啊闺属,到處亂用~

function sleep(second) {
    return new Promise((resolve, reject) => {
        setTimeout(() => {
            resolve('request done! ' + Math.random());
        }, second);
    })
}

async function bugDemo() {
    await sleep(1000);
    await sleep(1000);
    await sleep(1000);
    console.log('clear the loading~');
}

bugDemo();

loading 確實(shí)是等待請求都結(jié)束完才清除的。但是你認(rèn)真的觀察下瀏覽器的 timeline 請求是一個(gè)結(jié)束后再發(fā)另一個(gè)的(若觀察效果請發(fā)真實(shí)的 ajax 請求)
那么周霉,正常的處理是怎樣的呢掂器?

async function correctDemo() {
    let p1 = sleep(1000);
    let p2 = sleep(1000);
    let p3 = sleep(1000);
    await Promise.all([p1, p2, p3]);
    console.log('clear the loading~');
}
correctDemo();// clear the loading~

恩, 完美俱箱」停看吧~ async-await并不能取代promise.

附上最近寫的一個(gè)mpvue的demo請求:

image.png

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子乃摹,更是在濱河造成了極大的恐慌禁漓,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,496評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件孵睬,死亡現(xiàn)場離奇詭異播歼,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)掰读,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,407評論 3 392
  • 文/潘曉璐 我一進(jìn)店門秘狞,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人蹈集,你說我怎么就攤上這事烁试。” “怎么了拢肆?”我有些...
    開封第一講書人閱讀 162,632評論 0 353
  • 文/不壞的土叔 我叫張陵减响,是天一觀的道長。 經(jīng)常有香客問我郭怪,道長辩蛋,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,180評論 1 292
  • 正文 為了忘掉前任移盆,我火速辦了婚禮悼院,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘咒循。我一直安慰自己据途,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,198評論 6 388
  • 文/花漫 我一把揭開白布叙甸。 她就那樣靜靜地躺著颖医,像睡著了一般。 火紅的嫁衣襯著肌膚如雪裆蒸。 梳的紋絲不亂的頭發(fā)上熔萧,一...
    開封第一講書人閱讀 51,165評論 1 299
  • 那天,我揣著相機(jī)與錄音僚祷,去河邊找鬼佛致。 笑死,一個(gè)胖子當(dāng)著我的面吹牛辙谜,可吹牛的內(nèi)容都是我干的俺榆。 我是一名探鬼主播,決...
    沈念sama閱讀 40,052評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼装哆,長吁一口氣:“原來是場噩夢啊……” “哼罐脊!你這毒婦竟也來了定嗓?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,910評論 0 274
  • 序言:老撾萬榮一對情侶失蹤萍桌,失蹤者是張志新(化名)和其女友劉穎宵溅,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體上炎,經(jīng)...
    沈念sama閱讀 45,324評論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡恃逻,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,542評論 2 332
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了反症。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片辛块。...
    茶點(diǎn)故事閱讀 39,711評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡畔派,死狀恐怖铅碍,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情线椰,我是刑警寧澤胞谈,帶...
    沈念sama閱讀 35,424評論 5 343
  • 正文 年R本政府宣布,位于F島的核電站憨愉,受9級特大地震影響烦绳,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜配紫,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,017評論 3 326
  • 文/蒙蒙 一舰罚、第九天 我趴在偏房一處隱蔽的房頂上張望膊毁。 院中可真熱鬧,春花似錦、人聲如沸吃型。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,668評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽打颤。三九已至,卻和暖如春于个,著一層夾襖步出監(jiān)牢的瞬間氛魁,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,823評論 1 269
  • 我被黑心中介騙來泰國打工厅篓, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留秀存,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,722評論 2 368
  • 正文 我出身青樓羽氮,卻偏偏與公主長得像应又,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子乏苦,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,611評論 2 353

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