minOS

자바 예외 이해 - 언체크 예외 활용 본문

TIL/김영한의 스프링 DB 1편

자바 예외 이해 - 언체크 예외 활용

minOE 2024. 12. 9. 17:00
728x90

언체크 예외 활용

런타임 예외 사용 변환 - 코드 - UncheckedAppTest
package hello.jdbc.exception.basic;

import lombok.extern.slf4j.Slf4j;
import org.assertj.core.api.Assertions;
import org.junit.jupiter.api.Test;

import java.net.ConnectException;
import java.sql.SQLException;

@Slf4j
public class UnCheckedAppTest {


    @Test
    void unchecked(){
        Controller controller = new Controller();
        Assertions.assertThatThrownBy(()->controller.request())
                .isInstanceOf(RuntimeSQLException.class);
    }


    @Test
    void printEx(){
        Controller controller = new Controller();
        try{
            controller.request();
        }catch (Exception e){
            log.info("ex",e);
        }
    }


    static class Controller{
        Service service = new Service();

        public void request()  {
            service.logic();
        }
    }

    static class Service {
        Repository repository = new Repository();
        NetworkClient networkClient  = new NetworkClient();

        public void logic() {
            repository.call();
            networkClient.call();
        }

    }

    static class NetworkClient {
        public void call()  {
            throw new RuntimeConnectException("연결 실패");
        }

    }

    static class Repository {
        public void call()  {

            try {
                runSQL();
            } catch (SQLException e) {
                throw new RuntimeSQLException(e);
            }

        }

        public void runSQL() throws SQLException {
            throw new SQLException("ex");
        }
    }

    static class RuntimeConnectException extends RuntimeException{
        public RuntimeConnectException(String message) {
            super(message);
        }
    }

    static class RuntimeSQLException extends  RuntimeException{


        public RuntimeSQLException(Throwable cause) {
            super(cause);
        }
    }

}

예외 전환
- `SQLException` 런타임 예외인 `RuntimeSQLException`으로 변환했다
     1) 참고로 이때 기존 예외를 포함해주어야 예외 출력시 스택 트레이스에서 기존 예외도 함께 확인할 있다.

- ConnectException` 대신에 `RuntimeConnectException` 사용하도록 바꾸었다.
     
`NetworkClient` 단순히 기존 체크 예외를 `RuntimeConnectException` 이라는 런타임 예외가 발생  하도록 코드를 바꾸었다.


-
런타임 예외이기 때문에 서비스, 컨트롤러는 해당 예외들을 처리할 없다면 별도의 선언 없이 그냥 두면 된다.




런타임 예외 - 대부분 복구 불가능한 예외

시스템에서 발생한 예외는 대부분 복구 불가능 예외이다. 런타임 예외를 사용하면 서비스나 컨트롤러가 이런 복구 불가

능한 예외를 신경쓰지 않아도 된다. 물론 이렇게 복구 불가능한 예외는 일관성 있게 공통으로 처리해야 한다.




런타임 예외 - 의존 관계에 대한 문제

런타임 예외는 해당 객체가 처리할 없는 예외는 무시하면 된다. 따라서 체크 예외 처럼 예외를 강제로 의존하지 않아

된다.


런타임 예외 throws 생략
 static class Controller{
        Service service = new Service();

        public void request()  {
            service.logic();
        }
    }

    static class Service {
        Repository repository = new Repository();
        NetworkClient networkClient  = new NetworkClient();

        public void logic() {
            repository.call();
            networkClient.call();
        }

    }​

 

런타임 예외이기 때문에 컨트롤러나 서비스가 예외를 처리할 수 없다면 다음 부분을 생략할 수 있다.
- `method() throws RuntimeSQLException, RuntimeConnectException`
-  따라서 컨트롤러와 서비스에서 해당 예외에 대한 의존 관계가 발생하지 않는다


런타임 예외 구현 기술 변경시 파급 효과

- 런타임 예외를 사용하면 중간에 기술이 변경되어도 해당 예외를 사용하지 않는 컨트롤러, 서비스에서는 코드를

경하지 않아도 된다.
-
구현 기술이 변경되는 경우, 예외를 공통으로 처리하는 곳에서는 예외에 따른 다른 처리가 필요할 있다. 하지만

공통 처리하는 한곳만 변경하면 되기 때문에 변경의 영향 범위는 최소화 된다.



런타임 예외는 문서화
- 런타임 예외는 문서화를 잘해야 한다.
- 또는 코드에 `throws 런타임예외` 을 남겨서 중요한 예외를 인지할 수 있게 해준다.

728x90