언어/Java

잘못된 값을 입력하면 예외를 발생시키고 입력 다시 받기

Octoping 2023. 11. 7. 14:54

우테코 3주차 미션 [로또]를 하다가 마주한 요구사항이었다.

 

사용자가 잘못된 값을 입력할 경우 `IllegalArgumentException`를 발생시키고, 에러 메시지를 출력 후 그 부분부터 입력을 다시 받는다.

 

 

예제와 함께 구현해보기

간단한 예제를 들어보자.

다음과 같은 함수들을 만들어야 한다고 하자.

  • 0 이상의 양수를 입력받는 getTryCount()
  • 1과 45 사이의 숫자를 중복 없이 6개 받고, 보너스 번호도 받는 getLottoNumber()
  • 그 외 등등 다른 input 함수들

 

간단히 만들어보기

getTryCount 만들어보기

일단 올바른 값이 들어올 때까지 무한해서 입력을 다시 받아야하니, while + if - break의 조합으로 만들어야 할 것이다.

그리고 잘못된 값이 발생하면 에외를 발생시켜야하므로 while문 안에 try-catch 문이 필요할 것이다.

 

public int getTryCount() {
    while(true) {
        try {
            String input = scanner.nextLine();

            validateIsNumber(input);
            validateIsPositive(input);

            return Integer.parseInt(input);
        } catch(IllegalArgumentException e) {
            System.out.println(e.getMessage());
        }
    }
}

코드가 이렇게 되겠다.

일단 아무 것도 안했는데 들여쓰기 중첩이 2개인게 벌써부터 화난다..

 

일단 다음 함수를 만들러 가자.

 

getLottoNumber 만들어보기

public LottoNumber getLottoNumber() {
    List<Integer> lottoNumber;
    while (true) {
        try {
            String input = scanner.nextLine();

            validateIsNumber(input);
            validateIsInRange(input);
            validateIsDuplicate(input);

            lottoNumber = mapToIntegerList(input);
        } catch(IllegalArgumentException e) {
            System.out.println(e.getMessage());
        }
    }

    int bonusNumber;
    while (true) {
        try {
            String input = scanner.nextLine();

            validateIsNumber(input);
            validateIsInRange(input);
            validateIsDuplicateWithLottoNumber(input, lottoNumber);

            bonusNumber = Integer.parseInt(input);
        } catch(IllegalArgumentException e) {
            System.out.println(e.getMessage());
        }
    }

    return new LottoNumber(lottoNumber, bonusNumber);
}

로또 숫자를 받고, 보너스 번호도 받아야 한다.

덕분에 while-try-catch가 2개나 들어왔다.

 

 

문제점

이와 같은 방식은 문제가 크게 2개 있다.

 

첫번째로, 입력을 받는 메소드들의 개수만큼 이 while - try - catch문이 계속 중복이 된다. 

두 번째로는 이 while문 때문에 실제 로직에 시선이 집중되지 않고 복잡하니 시선이 분산된다.

 

이 while-try-catch 문의 코드 중복을 처리하고 싶다는 욕망이 강하게 든다.

그런데, 이 구문의 안에 있는 '함수'는 어떻게 해야 추출할 수 있을까?

 

 

while-try-catch문 추출해서 중복 제거하기

우리는 while-try-catch문을 이용해서, 제대로 된 값을 얻을 수 있을 때까지 무한 루프를 돌도록 구현했다.

즉, while-try-catch문 안에 있는 함수는 '값을 제공하는 함수'이다.

 

그러므로 이 안의 함수를 Supplier를 통해 치환할 수 있을 것같다.

 

private <T> T doLoop(Supplier<T> inputFunction) {
    while (true) {
        try {
            return inputFunction.get();
        } catch (IllegalArgumentException e) {
            System.out.println(e.getMessage());
        }
    }
}

파라미터로 온 값이 예외를 발생하지 않을 때까지 무한 루프를 돌다가 값을 반환하는 함수를 만들어냈다.

 

 

getTryCount 다시 만들기

우리가 만든 doLoop을 이용해서 코드를 다시 만들어보자.

 

public int getTryCount() {
    return doLoop(() -> {
        String input = scanner.nextLine();

        validateIsNumber(input);
        validateIsPositive(input);

        return Integer.parseInt(input);
    });
}

확실히 좀 깔끔해진 것 같다.

기존에는 while-try-catch라는 매서운 녀석들 때문에 로직에 온전히 집중이 되지 않고 시선이 분산되었는데, 이제는 딱 로직에 집중할 수있게 되었다.

 

 

getLottoNumber 다시 만들기

얘도 다시 만들어보자.

public LottoNumber getLottoNumber() {
    List<Integer> lottoNumber = doLoop(() -> {
        String input = scanner.nextLine();

        validateIsNumber(input);
        validateIsInRange(input);
        validateIsDuplicate(input);

        return mapToIntegerList(input);
    });
    
    int bonusNumber = doLoop(() -> {
        String input = scanner.nextLine();

        validateIsNumber(input);
        validateIsInRange(input);
        validateIsDuplicateWithLottoNumber(input, lottoNumber);

        return Integer.parseInt(input);
    });

    return new LottoNumber(lottoNumber, bonusNumber);
}

 

좋다.