디케이
예외처리 try catch 본문
1. 예외 처리
프로그램을 만들다 보면 수많은 에러를 만나기도 하지만 문법적인 오류가 아닌 예외 사항이 발생하기도 한다. 이때, 이에 대한 적절한 처리가 필요한데 간단한 예외의 경우 if 조건문으로 처리할 수 있다.
하지만 이렇게 되면 정상적인 if 코드와 구분이 어렵다. 이럴 때 사용할 수 있는 게 바로 try catch다.
try catch의 형태는 아래와 같다.
try {
예외가 발생할 구문
} catch (Exception e) {
처리할 구문
}
try의 안에는 예외가 발생할만한 부분을 넣어주고 catch의 괄호 안에 해당하는 예외와 같은 예외가 발생하면 try 구문을 더 실행하지 않고 catch로 넘어가 중괄호 안의 구문을 실행한다.
자바 컴파일러는 예외가 발생하는 순간 그 예외를 던지는데, chatch가 그 예외를 잡아 처리해 주는 역할을 하는 것이다.
try catch는 try catch finally의 형태로도 쓸 수 있는데, 이 형태는 try catch 내 예외 발생 여부와 상관없이 반드시 실행해야 할 구문이 있을 때 사용한다.
try {
} catch(Exception e) {
} finally {
마지막에 출력되는 실행문
}
위의 형태를 사용하면 예외 처리를 위해 catch 구문을 실행하든 하지 않든 반드시 출력된다.
이제 try catch를 이용하여 위의 소스 코드를 변경해보자.
추가로 finally까지 구현해 보자.
catch의 내용을 실행하고도 finally의 출력문이 실행되었다.
이번엔 try 부분에 다른 출력 소스들을 추가해 보았다. 하지만 실행된 소스는 catch와 finally뿐이다. 이로써 try에서 예외가 발생하면 그다음 코드를 실행하지 않고 바로 catch로 넘어간다는 것을 알 수 있다.
그렇다면 소스가 정상적으로 출력되었을 때는 어떨까?
try 안의 출력문이 정상적으로 출력되었을 뿐만 아니라 finally의 출력문도 함께 실행되었다.
이처럼 finally에는 예외 발생 여부와 상관없이 반드시 실행되어야 하는 부분을 넣으면 된다.
2. 커스텀 익셉션 (Custom Exception)
커스텀 Exception은 Exception 클래스에 명시되지 않은 예외를 처리하고 싶을 때 사용할 수 있다.
간단히 상속의 흐름을 알아보자면 예외 처리를 하기 위한 최상위 클래스인 Throwable 클래스가 있고, 이를 상속받는 Error 클래스와 Exception 클래스가 있다.
Exception 클래스는 모든 예외를 처리할 수 있는 코드들을 담고 있고 Error 클래스는 모든 오류들을 담고 있는 클래스라고 할 수 있다.
때문에 커스텀 예외 처리를 위해서는 새로운 클래스를 생성하고 Exception 클래스를 상속받아야 한다.
예를 들어 나이를 출력하는 프로그램이 있다고 하자. 나이는 음수가 될 수 없기 때문에 오류가 발생해야 하는데 Exception 클래스 내에는 정의되어 있지 않다. 그렇기 때문에 직접 예외를 발생시키는 것이다.
위 코드에서 낯선 부분이 두 가지 정도 보인다.
1. throw
2. throws 예외명
간단히 말하자면 throw는 내가 직접 발생시킨 예외를 던지는 행위에 해당한다.
Exception에 명시된 예외는 JVM이 인식하여 내부적으로 처리하지만 커스텀 익셉션은 그렇지 못하다. 그렇기 때문에 사용자가 직접 예외를 생성하여 던져 주어야 한다.
메소드 선언문 옆 throws 예외명은 발생한 예외를 처리하지 않겠다고 선언하는 것이다.
custom 메소드에서 처리되지 못한 예외는 해당 메소드가 호출된 곳으로 돌아가면서 함께 던져지게 된다.
돌아간 부분이 try catch 형식 안이기 때문에 던져진(throw) 예외가 잡힐(catch) 수 있는 것이다.
쉽게 말해 custom 메소드는 자신의 예외를 메인 메소드에게 미룬 것과 같다.
만약 메인 메소드에도 throws AgeException가 선언되어 있었다면 예외는 처리되지 못하고 결국 프로그램은 예외 메시지를 띄우며 강제 종료된다.
그렇기 때문에 예외를 던지면 잡거나(catch) 받지 않겠다 명시(throws 예외명)하거나 둘 중 하나는 반드시 해야 한다.
3. 가볍게 읽어볼 내용
1. getMessage는 예외 상황을 알리기 위해 정의된 모든 예외 클래스들이 상속하는 Throwable 클래스에 정의되어 있는 메소드이다. 모든 예외들은 Throwable 클래스를 상속받는다.
2. Exception 종류
>> ArrayIndexOutOfBoundsException : 배열에 잘못된 인덱스 값을 사용하는 예외
>> ClassCastException : 허용되지 않는 형 변환을 진행하는 예외
>> NegativeArraySizeException : 배열의 크기를 음수로 지정하는 예외
>> NullPointerException : 참조 변수가 null인 상태에서 메소드를 호출할 때 발생하는 예외
3. try catch에서 두 개 이상의 catch를 사용할 경우 상위 클래스가 먼저 올 수 없다.
// 불가능한 예시 try { ... } catch(Throwable e) { // 가장 최상위 클래스이기 때문에 아래 예외가 실행될 수 X ... } catch(ArithmeticException e) { ... } // 가능한 예시 try { ... } catch(ArithmeticException e) { ... } catch(Throwable e) { ... }
4. throw는 예외 상황이 발생되었음을 자바 가상머신에게 알리는 키워드다. 따라서 이 문장이 실행되면서 자바의 예외 처리 메커니즘이 동작하게 된다.
5. Throwable 클래스에 정의되어 있는 printStackTrace 메소드로 예외가 발생하여 전달되는 과정을 출력할 수 있다.
6. 가상머신의 예외 처리 방식
>> 1) getMessage 메소드를 호출한다.
>> 2) 예외 상황이 발생하여 전달되는 과정을 출력해 준다. (printStackTrace 메소드)
>> 3) 프로그램을 종료한다.
7. Throwable을 상속하는 예외 클래스는 Error와 Exception 두 가지이다.
그런데 Error는 이름처럼 단순한 예외로 볼 수 없는 심각한 오류 상황을 표현하기 위해 정의된 클래스이다. 따라서 이 클래스를 상속하여 정의된 클래스는(프로그래머가 정의하는 클래스 X) 프로그램의 실행을 멈춰야 할 정도로 매우 심각한 오류 상황을 표현하는 데 사용된다.
8. Error 클래스를 상속하는 대표적인 클래스는 VirtualMachineError이다. API 문서에서는 이 클래스를 "자바 가상머신에 문제가 생겨 더 이상 제대로 동작할 수 없는 상황을 알리기 위해 정의된 클래스이다."라고 설명한다. 그렇기 때문에 Error를 상속하는 클래스의 오류 상황이 발생하면, 그냥 프로그램이 종료되도록 놔두는 것이 상책이다. 프로그램이 종료된 이후 소스코드를 수정하는 등의 방식으로 원인을 해결해야 한다.
9. VirtualMachineError 클래스를 상속하는 클래스 중 OutOfMemoryError라는 클래스가 있는데, 이는 메모리 공간이 부족한 상황을 표현하는 예외 클래스이다. 따라서 이러한 오류가 발생하면, 메모리가 효율적으로 사용되도록 소스코드 자체를 변경해야 한다.
10. Exception을 상속하는 대표적 클래스는 두 가지이다.
>> RuntimeException : 주로 프로그램의 오류가 있을 때 발생하도록 의도된 것들(NullPointer 등과 같이 부주의로 생기는 오류), 그렇기 때문에 try catch문을 사용하지 않고도 예외를 처리할 수 있다.
(다만, 프로그램의 실행을 이어가기 위해 try catch문으로 해당 예외를 처리하기도 한다)
>> IOException : 입출력과 관련된 오류, 반드시 try catch문으로 예외를 처리해야 한다.
11. Exception을 상속하는 클래스의 예외 상황이 임의의 메소드 내에서 발생한다면 해당 메소드는 반드시 두 가지 중 한 가지 방법을 선택하여 정의해야 한다.
>> 1) try catch문을 이용하여 메소드 내의 예외 처리
>> 2) trows를 이용하여 메소드를 호출한 영역으로 예외가 전달될 수 있도록 정의
'Java' 카테고리의 다른 글
프로그래머스 - null 처리(IF문) (0) | 2020.11.23 |
---|---|
에러 ) generated keys not requested. you need to specify statement.return_generated_keys to statement.executeupdate() (0) | 2020.11.18 |
DAO && DTO(=VO) 개념 (0) | 2020.11.03 |
2020 10 30 테스트 문제 (0) | 2020.10.30 |
git으로 이전 작업 불러오기, 원복 시키기 (0) | 2020.10.29 |