async-await使用方法


async-await和Promise的關系
  經常會看到有了 async-await、promise 還有必要學習嗎、async await優於promise的幾個特點,接收了這些信息后,就蒙圈了。現在才知道,async-await是promise和generator的語法糖。只是為了讓我們書寫代碼時更加流暢,當然也增強了代碼的可讀性。簡單來說:async-await 是建立在 promise機制之上的,並不能取代其地位。

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

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

 

  上述代碼就是async-await的基本使用形式。有兩個陌生的關鍵字async、await,同時函數執行結果似乎返回了一個promise對象。

async
async用來表示函數是異步的,定義的函數會返回一個promise對象,可以使用then方法添加回調函數。

async function demo01() {
  return 123;
}

demo01().then(val => {
console.log(val);// 123
});
  若 async 定義的函數有返回值,return 123;相當於Promise.resolve(123),沒有聲明式的 return則相當於執行了Promise.resolve();


await
  await 可以理解為是 async wait 的簡寫。await 必須出現在 async 函數內部,不能單獨使用。

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


  await 后面可以跟任何的JS 表達式。雖然說 await 可以等很多類型的東西,但是它最主要的意圖是用來等待 Promise 對象的狀態被 resolved。如果await的是 promise對象會造成異步函數停止執行並且等待 promise 的解決,如果等的是正常的表達式則立即執行。

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~
希望通過上面的 demo,大家可以理解我上面的話。

實例
  舉例說明啊,你有三個請求需要發生,第三個請求是依賴於第二個請求的解構第二個請求依賴於第一個請求的結果。若用 ES5實現會有3層的回調,若用Promise 實現至少需要3個then。一個是代碼橫向發展,另一個是縱向發展。今天指給出 async-await 的實現哈~

//我們仍然使用 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
錯誤處理
上述的代碼好像給的都是resolve的情況,那么reject的時候我們該如何處理呢?

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 的情況我們應該將代碼塊用 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 回來的數據了。


小心你的並行處理!!!
  我這里為啥加了三個感嘆號呢~,因為對於初學者來說一不小心就將 ajax 的並發請求發成了阻塞式同步的操作了,我就真真切切的在工作中寫了這樣的代碼。await 若等待的是 promise 就會停止下來。業務是這樣的,我有三個異步請求需要發送,相互沒有關聯,只是需要當請求都結束后將界面的 loading 清除掉即可。
剛學完 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 確實是等待請求都結束完才清除的。但是你認真的觀察下瀏覽器的 timeline 請求是一個結束后再發另一個的(若觀察效果請發真實的 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.

await in for 循環
最后一點了,await必須在async函數的上下文中的。

// 正常 for 循環
async function forDemo() {
  let arr = [1, 2, 3, 4, 5];
  for (let i = 0; i < arr.length; i ++) {
    await arr[i];
  }
}
forDemo();//正常輸出


// 因為想要炫技把 for循環寫成下面這樣
async function forBugDemo() {
  let arr = [1, 2, 3, 4, 5];
  arr.forEach(item => {
    await item;
  });
}
forBugDemo();// Uncaught SyntaxError: Unexpected identifier


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM