본문 바로가기

이것저것 스터디📚/JavaScript - Deep Dive

DeepDive - 47장. 에러 처리

47.1 에러 처리의 필요성

- 에러에 대해 대처하지 않고 방치하면 프로그램은 강제 종료된다.

- try...catch 문을 사용해 발생한 에러에 적절하게 대응하면 프로그램이 강제 종료되지 않고 계속해서 코드를 실행시킬 수 있다.

- 우리가 작성하는 코드에서는 언제나 에러나 예외적인 상황이 발생할 수 있다는 것을 전제하고 이에 대응하는 코드를 작성하는 것이 중요하다.


47.2 try... catch... finally 문

- try... catch... finally 문은 일반적인 에러 처리 방법이라고 한다.

- finally 문은 불필요하다면 생략 가능하다. catch 문도 생략이 가능하지만, catch문 없는 try 문은 의미가 없으므로 생략하지 않는다.

console.log('[Start]');

try {
  // 실행할 코드(에러가 발생할 가능성이 있는 코드)
  foo();
} catch (err) {
  // try 코드 블록에서 에러가 발생하면 이 코드 블록의 코드가 실행된다.
  // err에는 try 코드 블록에서 발생한 Error 객체가 전달된다.
  console.error(err); // ReferenceError: foo is not defined
} finally {
  // 에러 발생과 상관없이 반드시 한 번 실행된다.
  console.log('finally');
}

// try...catch...finally 문으로 에러를 처리하면 프로그램이 강제 종료되지 않는다.
console.log('[End]');

47.3 Error 객체

- Error 생성자 함수는 에러 객체를 생성하며, Error 생성자 함수에는 에러를 상세히 설명하는 에러 메시지를 인수로 전달할 수 있다.

- Error 생성자 함수가 생성한 에러 객체는 message 프로퍼티와 stack 프로퍼티를 갖는다.

- message 프로퍼티 : Error 생성자 함수에 인수로 전달한 에러 메시지

- stack 프로퍼티 : 에러를 발생시킨 콜 스택의 호출 정보를 나타내는 문자열

 

- 자바스크립트는 7가지의 에러 객체를 생성할 수 있는 Error 생성자 함수를 제공한다.

- 아래의 에러 생성자 함수가 생성한 에러 객체의 프로토타입은 모두 Error.prototype을 상속받는다.

생성자 함수 인스턴스
Error 일반적인 에러 객체
SyntaxError 자바스크립트 문법에 맞지 않는 문을 해석할 때 발생하는 에러 객체
ReferenceError 참조할 수 없는 식별자를 참조했을 때 발생하는 에러 객체
TypeError 피연산자 또는 인수의 데이터 타입이 유효하지 않을 때 발생하는 에러 객체
RangeError 숫자값의 허용 범위를 벗어났을 때 발생하는 에러 객체
URIError encodeURI 또는 decodeURI 함수에 부적절한 인수를 전달했을 때 발생하는 에러 객체
EvalError eval 함수에서 발생하는 에러 객체

474. throw 문

- Error 생성자 함수로 에러 객체를 생성한다고 에러가 발생하는 것은 아니다. 즉, 에러 객체 생성과 에러 발생은 의미가 다르다.

- 에러를 발생시키려면 try 코드 블록에서 throw 문으로 에러 객체를 던져야 한다.

try {
  // 에러 객체를 생성한다고 에러가 발생하는 것은 아니다.
  new Error('something wrong');
} catch (error) {
  console.log(error);
}

- throw 문의 표현식은 어떤 값이라도 상관없지만 일반적으로 에러 객체를 지정하고, 에러를 던지면 catch의 에러 변수가 생성되고 던져진 에러 겍체가 할당된다.

// 외부에서 전달받은 콜백 함수를 n번만큼 반복 호출한다
const repeat = (n, f) => {
  // 매개변수 f에 전달된 인수가 함수가 아니면 TypeError를 발생시킨다.
  if (typeof f !== 'function') throw new TypeError('f must be a function');

  for (var i = 0; i < n; i++) {
    f(i); // i를 전달하면서 f를 호출
  }
};

try {
  repeat(2, 1); // 두 번째 인수가 함수가 아니므로 TypeError가 발생(throw)한다.
} catch (err) {
  console.error(err); // TypeError: f must be a function
}

47.5 에러의 전파

- 에러는 호출자 방향으로 전파된다. 즉, 콜 스택의 아래 방향(실행 중인 실행 컨텍스트가 푸시되기 직전에 푸시된 실행 컨텍스트 방향)으로 전파된다.

const foo = () => {
  throw Error('foo에서 발생한 에러'); // ④
};

const bar = () => {
  foo(); // ③
};

const baz = () => {
  bar(); // ②
};

try {
  baz(); // ①
} catch (err) {
  console.error(err);
}

- 위 예제에서, baz -> bar -> foo 함수 순서로 호출되고 foo 함수는 ④에서 에러를 throw 한다.

- 이때 foo 함수가 throw한 에러는 다음과 같이 호출자에게 전파되어 전역에서 캐치된다.

- 비동기 함수인 setTimeout이나 프로미스 후속 처리 메서드의 콜백 함수는 호출자가 없다는 것을 주의해야 한다.

- setTimeout이나 프로미스 후속 처리 메서드의 콜백 함수는 테스크 큐나 마이크로태스크 큐에 일시 저장되었다가 콜 스택이 비면 이벤트 루프에 의해 콜 스택으로 푸시되어 실행된다.

- 이때 콜 스택에 푸시된 콜백 함수의 실행 컨텍스트는 콜 스택의 가장 하부에 존재하게 되며, 에러를 전파할 호출자가 존재하지 않는다.


이웅모 님의〔 모던 자바스크립트 Deep Dive 〕책을 공부하며 정리했습니다.

'이것저것 스터디📚 > JavaScript - Deep Dive' 카테고리의 다른 글

DeepDive - 48장. 모듈  (0) 2023.10.17
DeepDive - 45장. 프로미스  (1) 2023.10.10
DeepDive - 44장. REST API  (0) 2023.09.26
DeepDive - 43장. Ajax  (0) 2023.09.26
DeepDive - 42장. 비동기 프로그래밍  (0) 2023.09.19