Programming Language

JS33 - 26.Async/Await

양찬우 2021. 10. 27. 15:46
728x90

Async/Await

async function 함수명() {
  await 비동기_처리_메서드_명();
}
  • async와 await는 자바스크립트의 비동기 처리 패턴 중 가장 최근에 나온 문법 (ES2017 ES8 )
  • 내부적으론 비동기 작업을 수행하면서, 표면적으론 완전히 동기식 코드를 작성할 수 있도록 도와준다.
  • 기존의 비동기 처리 방식인 콜백 함수와 프로미스의 단점을 보완하고 개발자가 읽기 좋은 코드를 작성할 수 있다.
  • 일반적으로 자바스크립트의 비동기 처리 코드는 아래와 같이 콜백을 사용해야 코드의 실행 순서를 보장받을 수 있다.
  • 함수 앞에 async가 붙으면 항상 promise를 리턴한다. 다른 값들은 자동적으로 resolve된 promise로 감싸진다.
// 미적용
function logName() {
  var user = fetchUser("domain.com/users/1");
  if (user.id === 1) {
    console.log(user.name);
  }
}

// 콜백처리
function logName() {
  // 아래의 user 변수는 위의 코드와 비교하기 위해 일부러 남겨놓았습니다.
  var user = fetchUser("domain.com/users/1", function (user) {
    if (user.id === 1) {
      console.log(user.name);
    }
  });
}

// async & await 적용 후
async function logName() {
  var user = await fetchUser("domain.com/users/1");
  if (user.id === 1) {
    console.log(user.name);
  }
}

사용법

  1. 함수 앞에 async 예약어를 붙인다.
  2. 함수의 내부 로직 중 HTTP 통신을 하는 비동기 처리 코드 앞에 await를 붙인다.
    • 비동기 처리 메서드가 꼭 프로미스 객체를 반환해야 await가 의도한 대로 동작합니다.
    • 일반적으로 await의 대상이 되는 비동기 처리 코드는 Axios 등 프로미스를 반환하는 API 호출 함수입니다.

작동방식

  1. async함수가 호출되면 Promise가 반환됩니다.
  2. async함수가 값을 반환하면 Promise는 반환 된 값으로 해결됩니다.
  3. async함수가 예외 또는 일부 값을 throw하면 Promise는 throw된 값과 함께 거부됩니다.
  • async함수는 await표현식을 포함 할 수 있으며, async함수의 실행을 일시 중지하고 전달된 Promises를 기다린 다음 async함수의 실행을 재개하고 해결 된 결과를 리턴합니다.

예외처리

  • try catch를 사용한다.
  • 프로미스에서 에러 처리를 위해 .catch()를 사용했던 것처럼 async에서는 catch {} 를 사용하면 된다.
  • 오류 처리가 완전히 동기적으로 수행된다.

Async

async function f() {
  return 1;
}
  • 함수 앞의 async 단어는 한가지 뜻을 지닌다: 이 함수는 항상 promise를 리턴합니다. 다른 값들은 자동적으로 resolve된 promise로 감싸집니다.
  • await 은 오직 async 함수 안에서만 동작한다.

Await

// works only inside async functions
let value = await promise;
  • await 키워드는 자바스크립트가 promise가 셋팅되고 그 결과 값이 리턴될 때까지 기다립니다.
  • await 은 문자 그대로 promise가 해결될 때까지 자바스크립트를 기다리게 만듭니다. 그리고 결과가 나오면 계속 진행이 됩니다. 이 작업은 CPU 비용(낭비)이 들지 않습니다. 왜냐하면 엔진은 그 동안에 다른 작업을 할 수 있습니다. : 다른 스크립트나 다른 이벤트를 다룰 수 있습니다.
  • 코드의 최상단에 await을 사용할 수 없다.
// syntax error in top-level code
let response = await fetch('/article/promise-chaining/user.json');
let user = await response.json();

// 대신 비동기 함수(async)로 감싸서 사용할 수 있다.
(async () => {
  let response = await fetch('/article/promise-chaining/user.json');
  let user = await response.json();
  ...
})();
  • promise.then 처럼, await는 then 메서드가 있는 객체를 사용할 수 있다.

서드파티 객체가 만약에 promise가 아니지만 promise 처럼 움직일수 있는: 만약 .then 을 지원하는 객체라면 충분히 await 을 사용할 수 있습니다.

  • 만약에 await이 .then 을 갖는 promise 객체가 아닐경우 그것은 네이티브 함수 resolve, reject 를 매개변수로 제공받을 수 있습니다. 그리고 await은 그중 하나가 호출 될때까지 기다립니다.
class Thenable {
  constructor(num) {
    this.num = num;
  }
  then(resolve, reject) {
    alert(resolve);
    // resolve with this.num*2 after 1000ms
    setTimeout(() => resolve(this.num * 2), 1000); // (*)
  }
}

async function f() {
  // waits for 1 second, then result becomes 2
  let result = await new Thenable(1); //Thenable이 .then을 지원한다.
  alert(result);
}

Async Class method 비동기 클래스 메소드

  • 비동기 클래스 메서드를 선언하기 위해서는 단지 async만 함수 앞에 붙여주면 됩니다.
class Waiter {
  async wait() {
    return await Promise.resolve(1);
  }
}

new Waiter().wait().then(alert); // 1
  • 반환받는 값이 promise이고 await 을 사용할 수 있다는 보장을 하는 것이다. 함수와 동일하다.

async/await 과 promise

async/await and promise.then/catch

  • 우리가 async/await을 사용할 때, .then이 거의 필요가 없다, 왜냐하면 await 는 우리를 위해 기다려주기 때문이다. 그리고 우리는 대게 .catch 대신에 try..catch를 사용하기 때문이다. 이 방법은 항상 그렇진 않을 수 있지만 대게 편리합니다. 하지만 코드 최상위에서 async 함수 밖에서 사용할 때 우리는 문법적으로 await을 사용할 수 없습니다. 그래서 우리는 최종 결과 또는 에러를 다루기 위해 .then/catch 문을 연습해야 한다.
async function f() {
  let response = await fetch("http://no-such-url");
}

// f() becomes a rejected promise
f().catch(alert); // TypeError: failed to fetch
// async 밖이기 때문에 문법적으로 await이 불가해서 fetch 못함

async/await works well with Promise.all

  • 여러 promise를 기다려야 할때, Promise.all 과 await 로 감쌀 수 있다.
// wait for the array of results
let results = await Promise.all([
  fetch(url1),
  fetch(url2),
  ...
]);


// 예시 2
async  function msg() {
	// array destructuring
	const [a, b, c] = await Promise.all([who(), what(), where()]);
	console.log(`${ a } ${ b } ${ c }`);
}

msg(); // 🤡 lurks in the shadows <-- after 500ms

Arrow function async

  • 화살표 함수를 사용해서 만들 수도 있다.
// 함수표현식 async-await
const msg = async function () {
  const msg = await scaryClown();
  console.log("Message:", msg);
};

// 화살표 함수 async-await
const msg = async () => {
  const msg = await scaryClown();
  console.log("Message:", msg);
};

참고

728x90