# [frontend] 在 async / await 中更好的處理錯誤 > 原文链接:Better error handling with async/await,by Sobio Darlington ### callback hell 了解 Promise 原理,先從 callback hell 開始,因為 callback 的嵌套,會使程式碼向右邊排不而不是垂直向下排版 ```javascript= // Code that reads from left to right // instead of top to bottom let user; let friendsOfUser; getUser(userId, function(data) { user = data; getFriendsOfUser(userId, function(friends) { friendsOfUser = friends; getUsersPosts(userId, function(posts) { showUserProfilePage(user, friendsOfUser, posts, function() { // Do something here }); }); }); }); ``` ### Promise 用來更好的處理異步問題,避免 callback hell 問題 ```javascript= // A solution with promises let user; let friendsOfUser; getUser().then(data => { user = data; return getFriendsOfUser(userId); }).then(friends => { friendsOfUser = friends; return getUsersPosts(userId); }).then(posts => { showUserProfilePage(user, friendsOfUser, posts); }).catch(e => console.log(e)); ``` ### async / await Promise Promise 的語法糖,讓 Promise 更簡潔的寫法 在 function 前面加上 async 就可以將 function 轉換成 Promise ```javascript= async function add(a, b) { return a + b; } add(1, 2).then(result => conosle.log(result)); ``` ### async / await Promise 遇到的情境,及優化方式 ```javascript= async function userProfile() { let user = await getUser(); let friendOfUser = await fetFriendOfUser(userId); let posts = await getUsersPosts(userId); showUserProfilePage(user, friendOfUser, posts); } ``` 在以上 如果有一個 Promise 失敗的話就會回傳一個 reject,會拋出 Unhandled promise rejection 異常。處理這問題比較容易的方式是在外層加一個 try...catch ```javascript= async function userProfile() { try { let user = await getUser(); let friendOfUser = await fetFriendOfUser(userId); let posts = await getUsersPosts(userId); showUserProfilePage(user, friendOfUser, posts); } catch (e) { console.log(e); } } ``` 但是無法知道是哪一個 Promise 發生錯誤,可以在異步請求上使用 catch ```javascript= async function userProfile() { let user = await getUser().catch(e => console.log(e.message)); let friendOfUser = await fetFriendOfUser(userId).catch(e => console.log(e.message)); let posts = await getUsersPosts(userId).catch(e => console.log(e.message)); showUserProfilePage(user, friendOfUser, posts); } ``` ### 更好的錯誤處理方式 ```javascript= /** * @description ### Returns Go / Lua like responses(data, err) * when used with await * * - Example response [ data, undefined ] * - Example response [ undefined, Error ] * * * When used with Promise.all([req1, req2, req3]) * - Example response [ [data1, data2, data3], undefined ] * - Example response [ undefined, Error ] * * * When used with Promise.race([req1, req2, req3]) * - Example response [ data, undefined ] * - Example response [ undefined, Error ] * * @param {Promise} promise * @returns {Promise} [ data, undefined ] * @returns {Promise} [ undefined, Error ] */ const handle = (promise) => { return promise .then(data => ([data, undefined])) .catch(error => Promise.resolve([undefined, error])); } async function userProfile() { let [user, userErr] = await handle(getUser()); if(userErr) throw new Error('Could not fetch user details'); let [friendsOfUser, friendErr] = await handle( getFriendsOfUser(userId) ); if(friendErr) throw new Error('Could not fetch user\'s friends'); let [posts, postErr] = await handle(getUsersPosts(userId)); if(postErr) throw new Error('Could not fetch user\'s posts'); showUserProfilePage(user, friendsOfUser, posts); } ``` 這樣只需要用到一個工具函數 handle,就可以避免 Unhandled promise rejection 報錯,還能細度處理錯誤。 - 如果 Promise resolve,handle 回傳 [data, undefined] - 如果 Promise reject,handle 回傳 [undefined, Error] [[译] 在 async/await 中更好的处理错误](https://juejin.im/post/6844904071124500488#heading-12)