코드를 완벽하게 짰다고 해서 항상 프로그램이 성공적으로 도는 것은 아니다. 다양한 예외 상황이 있을 발생할 수 있는데, 이것에 대응하기 위한 예외 처리 코드가 필요하다.
예외 처리의 목적
1. 예외의 발생으로 인한 실행 중인 프로그램의 비정상 종료를 막기 위함.
2. 개발자에게 알려서 코드를 보완할 수 있도록 하기 위함.
Java에서는 상속을 이용해서 모든 예외를 표현한다. 모든 예외 클래스는 Throwable
의 자식 클래스다. Throwable
에는 크게 두 종류의 자식 클래스가 존재한다.
Error
는 프로그램이 종료되어야 하는 심각한 문제를 표현한다. 대부분 컴퓨터나 JVM이 시스템적으로 동작할 수 없는 상황을 표현한다.Exception
은 프로그램이 종료되지는 않지만 예외나 문제 상황을 표현하기 위해서 사용된다.Java는 JVM내의 Heap이라는 메모리 공간을 OS로부터 할당 받아 사용한다. 할당 받을 수 있는 최대 메모리 이상을 사용하면, JVM이 다운될 수 밖에 없다. 이 경우
OutOfMemoryError
가 나면서 프로그램이 종료된다. Java의 대표적인 에러 상황을 줄여서OOM
이라고 한다.
대부분 표현하려는 예외 상황은 대부분 Exception
종류일 것이다. 실행도중 발생하는 Exception은 RunTimeException
을 상속받아서 정의를 한다.
파일을 읽고 쓰거나, 원격에 있는 저장소로부터 데이터를 읽고 쓸 때 나는 에러를 표현하려면 IOException
을 상속받아서 정의한다.
public class Exception {
public static void main(String[] args) {
int number = 10;
int result;
for(int i = 10; i >= 0; i--) {
try {
result = number / i;
System.out.println(result);
} catch (java.lang.Exception e) {
System.out.println("Exception 발생: " + e.getMessage());
} finally {
System.out.println("항상 실행되는 finally 구문");
}
}
}
}
try-catch(-finally)
형식을 이용하여 예외 처리를 진행했다. finally
구문은 필수는 아니며, 예외 발생여부에 관계 없이 항상 수행되어야하는 코드를 구현한다.
만약에 예외가 발생하지 않는다면 try -> finally
순으로 실행이된다. 또한, 예외는 중복 catch
블럭을 사용하여 다양한 예외 처리를 수행할 수 있다.
중복 catch
블럭을 사용할 때는 먼저 선언된 catch
블럭부터 확인한다. 앞의 catch
블럭에서 잡혔다면, 뒤에 catch
블럭으로는 전파되지 않는다.
좁은 범위의 예외부터 앞에서 선언하는 것이 좋다. 여기서 좁은 범위는 상속관계에서 자식 클래스에 위치 할수록 좁은 법위다.
예를 들어 IOException
이 발생할 것 같아 예외 처리를 하고, 그 외의 예외 처리를 하고 싶다면 IOException
을 catch
하는 구문을 먼저, Exception
을 catch
하는 구문을 그 뒤에 작성한다.
try-catch
문 외에 try-with-resource
문도 존재한다. 입출려과 함께 쓰이는 구문이며, 일반적으로 사용됐던 자원을 끝난 후에 닫아줘야하는 것들이 존재하는데 try-with-resource
문이 있다.
기존에 try-catch(-finally)
문은 자원을 닫을 때 close()
를 사용해야한다. try-with-resource
문은 try
를 벋어나는 순간 자동으로 close()
가 호출된다.
사용방법은 try()
안의 입출력 스트림을 생성하는 로직을 작성할 때 해당 객체가 AutoClosable
인터페스를 구현한 객체여야 한다.
import java.io.FileOutputStream;
import java.io.IOException;
public class Exception {
public static void main(Stirng[] args) {
try (FileOutputStream out = new FileOutputStream("test.txt")) {
// test.txt file에 Hello World를 출력
out.write("Hello World".getBytes());
out.flush();
} catch (IOException e) {
e.printStackTrace();
}
}
}
이처럼 형식은 try-catch
문과 비슷하지만, try()
안에 AutoClosable
인터페이스를 구현한 객체를 선언하면 사용할 수 있다.
AutoClosable 인터페이스를 사용해야하는 이유는 AutoClosable 인터페이스에는 예외가 발생할 경우
close()
메서드를 호출하기로 정의되어 있기 때문이다.
만약 try-catch
문을 사용한다면 아래와 같은 코드가 된다.
import java.io.FileOutputStream;
import java.io.IOException;
public class Exception {
public static void main(String[] args) throws IOException{
FileOutputStream out = new FileOutputStream("test.txt");
try {
// test.txt file에 Hello World 출력
out.write("Hello World".getBytes());
out.flush();
} catch (IOException e) {
e.printStackTrace();
}
out.close();
}
}
위 코드처럼 코드가 길어질 뿐만 아니라 FileOutputStream
을 열고 닫을 떄 생기는 Exception
까지 그 상위에서 catch
를 하거나 throws
로 감싸줘야 한다.
catch
문을 이용해서 예외 처리를 하지 않은 경우, 메서드에 throws
로 예외가 발생할 수 있는 것을 알려줘야 한다. throws
키워드가 있는 함수를 호출하면, caller
쪽에서 catch
와 관련된 코드를 작성해줘야 한다.
void method() throws IndexOutOfBoundsException, IllegalArgumentException {
// method 내용
}
이처럼 키워드 throws
를 사용해서 메서드 내에서 발생할 수 있는 예외를 적어주면 된다.