📝 프로그램 오류
- 프로그램이 실행 중 어떤 원인에 의해서 오작동을 하거나 비정상적으로
종료되는 경우 이러한 원인을 프로그램 '에러' 또는 '오류'라고 한다.
- 이를 발생시점에 따라 '컴파일 에러'와 '런타임 에러'로 나눌 수 있다.
소스코드의 구문 오류로 인하여 컴파일이 불가능한 상태
이클립스에서 빨간색으로 표시되는 경우 해당한다.
프로그램을 실행하기 전에 발견되므로 상대적으로 고치기 쉽다.
구문상의 에러는 없지만, 프로그램이 실행되는 과정에서 다양한
경우의 수에 대응하지 못하여 발생하는 예외상황
런타임 에러가 발생하면 프로그램은 강제로 종료된다.
우리가 프로그램의 에러라고 부르는 현상은 대부분 런타임 에러에
해당한다.
프로그램 코드에 의해서 수습될 수 있는 다소 미약한 오류
자바 가상 머신은 프로그램 실행중에 예외가 발생하면
관련된 예외 클래스로부터 예외 객체를 생성하여 프로그램에서 지정된
예외처리 구문으로 넘긴다.
예외가 발생하면 프로그램에 지정된 예외 처리 구문이 자바 가상머신에
의해 호출되고, 이때 전달되는 예외 객체를 넘겨받아 적절한 처리를
수행한다.
예외처리란?
: 프로그램 실행 시 발생할 수 있는 예기치 못한 예외의 발생에
대비한 코드를 작성하는 것이다.
예외처리의 목적
: 프로그램의 비정상 종료를 막고, 정상적인 실행상태를 유지할 수
있도록 하는 것이다.
try{
// 기본적으로 실행되는 부분
} catch (예외클래스이름 객체이름) {
// try 블록에서 예외가 발생한 경우 이 곳으로 제어가 넘어온다.
} finally {
// 예외 상황의 발생 여부에 상관 없이 무조건 실행되는 부분
// finally 블록은 생략 가능하다.
}
try 블록 실행도중 catch에서 명시한 예외 클래스에 해당하는 런타임 에러가
발생하면, try 블록을 즉시 중단하고 catch 블록이 실행 된 후, finally 블록이
실행된다.
예외가 발생하지 않을 경우에는 try 블록 종료 후 finally 블록이 실행된다.
try{
//
} catch (NumberFormatException e) {
} catch (ArrayIndexOutOfBoundsException e) {
}
public class Main {
public static void main(String[] args) {
String year = "두 살";
int age = 2022 - Integer.parseInt(year) + 1;
System.out.println(age);
System.out.println("--------프로그램 종료--------");
}
}
👉 실행 결과
문자열은 숫자로 변환할 수 없으므로, 에러가 발생한다.
public class Main {
public static void main(String[] args) {
try {
String year = "두 살";
int age = 2022 - Integer.parseInt(year) + 1;
System.out.println(age);
} catch (NumberFormatException e) {
System.out.println("에러가 발생했습니다.");
}
System.out.println("--------프로그램 종료--------");
}
}
👉 실행 결과
에러가 발생했습니다.
--------프로그램 종료--------
try ~ catch 블록으로 예외처리가 적용된 경우에는, 에러가 발생하더라도
프로그램 자체가 다운되지는 않는다.
예외 종류를 의미하는 모든 클래스는 java.lang.Exception 클래스를 상속받는다.
Java에서 예외 상황을 의미하는 모든 클래스들의 최상위 클래스
Exception 클래스로 catch 블록을 구성하면, 모든 예외 상황에 일괄적으로 대응할 수 있지만,
catch 블록이 세분화된 경우와는 달리 상황별 개별적인 처리는 불가능하다.
Exception 클래스에 대한 예외처리는 맨 마지막 catch 블록에 명시하여
'마지막 알 수 없는 에러'를 의미하도록 구성한다.
try{
//
} catch (NumberFormatException e) {
} catch (ArrayIndexOutOfBoundsException e) {
} catch (Exception e) {
}
printStackTrace() 또는 getMessage()와 같은 메서드를 통해서
예외의 발생원인을 알 수 있다.
printStackTrace()
getMessage()
public class Main {
public static void main(String[] args) {
try {
String year = "두 살";
int age = 2022 - Integer.parseInt(year) + 1;
System.out.println(age);
} catch (NumberFormatException e) {
System.out.println("에러가 발생했습니다.");
System.out.println("원인 : " + e.getMessage());
// 개발자가 보려는 용도로 출력하는 시스템 에러 메시지
e.printStackTrace();
}
System.out.println("--------프로그램 종료--------");
}
}
👉 실행결과
이 결과는 비정상적으로 종료되었을 때의 결과와 비슷하지만
예외는 try-catch문에 의해 처리되었으며 프로그램은 정상적으로 종료되었다.
throw 키워드를 사용해서 프로그래머가 고의로 예외를 발생시킬 수 있다.
예외 발생시키는 방법
Exception e = new Exception("고의로 발생시킴");
throw e;
public class Main {
public static void main(String[] args) {
try {
Exception e = new Exception("고의로 발생시킴");
throw e;
} catch (Exception e) {
System.out.println("에러 메시지 : " + e.getMessage());
e.printStackTrace();
}
System.out.println("-----프로그램 정상 종료-----");
}
}
👉 실행결과
Exception인스턴스를 생성할 때, 생성자에 String을 넣어주면,
이 String이 Exception인스턴스에 메시지로 저장된다.
이 메시지는 getMessage()를 이용해서 얻을 수 있다.
리턴타입 메서드명() throws Exception1, Exception2 .. ExceptionN {
// 메서드 내용
}
리턴타입 메서드명() throws Exception{
// 메서드 내용
}
public class Example {
public void method(String year) throws NumberFormatException{
int age = 2022 - Integer.parseInt(year) + 1;
System.out.println(age);
}
}
// Main 클래스
public class Main {
public static void main(String[] args) {
Example ex = new Example();
try {
ex.method("천구백오십년"); // 메서드 호출
} catch (NumberFormatException e) {
System.out.println("입력값이 숫자가 아닙니다.");
}
}
}
👉 실행결과
입력값이 숫자가 아닙니다.
throws 키워드가 붙어있는 method()메서드를 사용할 때
NumberFormatException 예외가 발생하면, 메서드를 사용한 곳에서
예외처리를 해야한다.
그래서 코드 예시에서 method()를 호출하는 main()메서드에서
try-catch문으로 예외 처리를 했다.
참고자료 : <Java의 정석>, <이것이 자바다>