[JAVA] 예외처리

:-)·2022년 5월 5일
0

☕ JAVA

목록 보기
13/15
post-thumbnail

📝 프로그램 오류

  • 프로그램이 실행 중 어떤 원인에 의해서 오작동을 하거나 비정상적으로
    종료되는 경우 이러한 원인을 프로그램 '에러' 또는 '오류'라고 한다.

  • 이를 발생시점에 따라 '컴파일 에러'와 '런타임 에러'로 나눌 수 있다.



👾 에러의 종류

  • 컴파일 에러

    • 소스코드의 구문 오류로 인하여 컴파일이 불가능한 상태

    • 이클립스에서 빨간색으로 표시되는 경우 해당한다.

    • 프로그램을 실행하기 전에 발견되므로 상대적으로 고치기 쉽다.


  • 런타임 에러

    • 구문상의 에러는 없지만, 프로그램이 실행되는 과정에서 다양한
      경우의 수에 대응하지 못하여 발생하는 예외상황

    • 런타임 에러가 발생하면 프로그램은 강제로 종료된다.

    • 우리가 프로그램의 에러라고 부르는 현상은 대부분 런타임 에러에
      해당한다.



런타임 에러의 종류

  • 논리 오류

    • 프로그램 자체의 구조적인 문제로 인한 오류로 논리적 모순이
      생기지 않도록 코드를 작성해야 한다.

  • 자바 가상머신 자체의 오류

    • 프로그래머가 책임질 수 있는 수준의 오류가 아니다.

  • 예외(Exception)

    • 프로그램 코드에 의해서 수습될 수 있는 다소 미약한 오류

    • 자바 가상 머신은 프로그램 실행중에 예외가 발생하면
      관련된 예외 클래스로부터 예외 객체를 생성하여 프로그램에서 지정된
      예외처리 구문으로 넘긴다.

    • 예외가 발생하면 프로그램에 지정된 예외 처리 구문이 자바 가상머신에
      의해 호출되고, 이때 전달되는 예외 객체를 넘겨받아 적절한 처리를
      수행한다.



예외처리 - (try ~ catch문)

  • 예외처리란?
    : 프로그램 실행 시 발생할 수 있는 예기치 못한 예외의 발생에
    대비한 코드를 작성하는 것이다.

  • 예외처리의 목적
    : 프로그램의 비정상 종료를 막고, 정상적인 실행상태를 유지할 수
    있도록 하는 것이다.


try ~ catch문

  • 예외를 처리하기 위해서는 try-catch문을 사용한다.

📌 try ~ catch문 구조

	try{
		// 기본적으로 실행되는 부분
    } catch (예외클래스이름 객체이름) {
		// try 블록에서 예외가 발생한 경우 이 곳으로 제어가 넘어온다.
	} finally {
		// 예외 상황의 발생 여부에 상관 없이 무조건 실행되는 부분
		// finally 블록은 생략 가능하다.
	}
  • try 블록 실행도중 catch에서 명시한 예외 클래스에 해당하는 런타임 에러가
    발생하면, try 블록을 즉시 중단하고 catch 블록이 실행 된 후, finally 블록이
    실행된다.

  • 예외가 발생하지 않을 경우에는 try 블록 종료 후 finally 블록이 실행된다.


catch블록을 사용한 복잡한 에러처리

  • catch 블록은 에러가 예상되는 상황에 대해 복수로 명시하는 것이 가능하다.
	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("--------프로그램 종료--------");
		
	}

}

👉 실행 결과

문자열은 숫자로 변환할 수 없으므로, 에러가 발생한다.

  • java 프로그램은 예외 상황을 만나면 그 즉시 실행을 중단하고 강제 종료 된다.
  • 때문에 이 프로그램의 맨 마지막에 출력하려는 메시지는 표시되지 않는다.



✍ 코드 예시 - try ~ catch문

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 클래스를 상속받는다.



Exception 클래스

  • Java에서 예외 상황을 의미하는 모든 클래스들의 최상위 클래스

  • Exception 클래스로 catch 블록을 구성하면, 모든 예외 상황에 일괄적으로 대응할 수 있지만,
    catch 블록이 세분화된 경우와는 달리 상황별 개별적인 처리는 불가능하다.

  • Exception 클래스에 대한 예외처리는 맨 마지막 catch 블록에 명시하여
    '마지막 알 수 없는 에러'를 의미하도록 구성한다.

	try{
    	//
	} catch (NumberFormatException e) {
	} catch (ArrayIndexOutOfBoundsException e) {
	} catch (Exception e) {
	} 



printStackTrace() & getMessage()

  • printStackTrace() 또는 getMessage()와 같은 메서드를 통해서
    예외의 발생원인을 알 수 있다.

  • printStackTrace()

    • 실제 예외 상황시에 출력되는 메시지를 강제로 출력한다.
    • 개발자가 catch 블록 안에서 예외 상황을 분석하기 위한 용도로 사용한다.
  • getMessage()

    • 간략한 에러 메시지를 리턴한다.
    • e.getLocalisMessage()도 같은 기능을 한다.

✍ 코드 예시)

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 키워드를 사용해서 프로그래머가 고의로 예외를 발생시킬 수 있다.

  • 예외 발생시키는 방법

    1. new 연산자를 이용해서 발생시키려는 예외 클래스의 객체를 만든다.
    Exception e = new Exception("고의로 발생시킴");
    1. throw 키워드를 이용해서 예외를 발생시킨다.
    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()를 이용해서 얻을 수 있다.



메서드에 예외 선언하기

  • 예외를 처리하는 방법에는 try-catch문을 사용하는 것 외에,
    예외를 메서드에 선언하는 방법이 있다.
    이때 사용하는 키워드는 throws이다.
  • throws 키워드는 메서드에서 처리하지 않은 예외를 호출한 곳으로
    떠넘기는 역할을 한다.

  • 메서드에 예외 선언하는 방법
    • throws 키워드를 메서드 선언부 끝에 작성한다.
    • 예외가 여러 개일 경우에는 쉼표( , )로 구분한다.
      리턴타입 메서드명() throws Exception1, Exception2 .. ExceptionN {
      		// 메서드 내용
      }
    • 아래와 같이 모든 예외의 최고조상인 Exception클래스를 메서드에
      선언하면, 모든 예외를 간단히 떠넘길 수도 있다.
      리턴타입 메서드명() 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문으로 예외 처리를 했다.


📙 throws 키워드 정리

  • throws 키워드가 붙어있는 메서드는 반드시 try 블록 내에서
    호출되어야 한다.
  • 그리고 catch 블록에서 떠넘겨 받은 예외를 처리해야 한다.

  • main() 메서드에서도 throws 키워드를 사용해서 예외를 떠넘길 수 있는데.
    결국 JVM이 예외의 내용을 console에 출력하는 것으로 예외 처리를 한다.
    • 하지만 main()메서드에서 throws Exception을 붙이는 것은
      사용자가 프로그램이 알 수 없는 예외 내용을 출력하고 종료되기 때문에
      좋지 못한 예외 처리 방법이다.
    • 그렇기 때문에 main()에서 try-catch문으로 최종 처리하는 것이
      가장 바람직하다.









참고자료 : <Java의 정석>, <이것이 자바다>

0개의 댓글