-
Notifications
You must be signed in to change notification settings - Fork 1.1k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
2단계 - 로또(자동) #4077
2단계 - 로또(자동) #4077
Conversation
추가: - InputView.java
추가: - LottoMachine.java - LottoNumber.java - LottoRound.java - LottoRank.java
추가: - Lotto.java - ResultView.java 변경: - LottoRound.java: 로또 번호 리스트에 대한 결과 응답 메소드 추가
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
안녕하세요 찬성님 😃
2단계 구현 잘해주셨네요 👍
몇 가지 코멘트 좀 남겨두었는데 확인해 주시고 다시 리뷰 요청해 주세요.
src/main/java/lotto/Lotto.java
Outdated
} | ||
|
||
public static void lotto() { | ||
int lottoCount = InputView.getPositiveNumberInput("구입금액을 입력해 주세요.") / 1000; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
로또 한 장은 1000원이다. 비즈니스 로직으로 볼 수 있지 않을까요?
적절한 도메인 객체에게 책임을 부여하면 어떨까요?
src/main/java/lotto/LottoConfig.java
Outdated
public class LottoConfig { | ||
public static final Integer MAX_LOTTO_NUMBER = 45; | ||
public static final int LOTTO_PRICE = 1000; | ||
public static final int LOTTO_NUMBER_SIZE = 6; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
각 상수가 필요한 객체 내부에 선언해 응집도를 높이면 어떨까요?
import static lotto.LottoConfig.MAX_LOTTO_NUMBER; | ||
|
||
public class LottoMachine { | ||
private static final List<Integer> LOTTO_NUMBERS = IntStream.range(1, MAX_LOTTO_NUMBER + 1).boxed().collect(Collectors.toList()); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
지난 단계에서도 피드백 드린 부분인데요
스트림을 활용하신다면 개행으로 가독성을 높여 보시면 좋겠습니다.
매직 넘버 '1' 도 상수로 치환하면 어떨까요?
src/main/java/lotto/Lotto.java
Outdated
public class Lotto { | ||
public static void main(String[] args) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
애플리케이션의 진입점이니 LottoMain, LottoApplication 같은 이름은 어떠신가요?
src/main/java/lotto/LottoNumber.java
Outdated
public class LottoNumber { | ||
private final Set<Integer> numbers; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
여러 개의 숫자를 다루니 클래스명을 복수로 표현하면 어떨까요?
long winnings = rankCount.entrySet().stream() | ||
.mapToLong(entry -> (long)entry.getKey().getMoney() * entry.getValue()) | ||
.sum(); | ||
|
||
System.out.printf("총 수익률은 %.2f입니다.\n", (double)winnings / (totalCount * LOTTO_PRICE)); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
당첨금 합산이나 수익률 계산은 비즈니스 로직으로 봐야하지 않을까요?
ResultView는 출력 자체에 집중하면 어떨까요?
import static org.junit.jupiter.api.Assertions.*; | ||
|
||
class LottoNumberTest { | ||
@DisplayName("LottoNumber 조건, 1~45 사이의 수 and 6개") |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
하나의 테스트에서 여러가지를 검증한다면 테스트 대상 메서드가 너무 많은 일을 하고 있는건 아닌지 고민해 볼 필요가 있습니다.
앞서 드린 코멘트처럼 번호의 범위에 대한 책임을 분리한다면 테스트도 간단해질 것 같아요.
Assertions.assertThat(LottoRank.valueOfMatchCount(2)) | ||
.isNull(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
null을 활용하면 언제나 NPE의 위험에 노출이 됩니다.
2개 이하 일치하는 '미당첨' 타입을 추가하면 어떨까요?
Assertions.assertThat(LottoRank.valueOfMatchCount(2)) | ||
.isNull(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
null을 활용하면 언제나 NPE의 위험에 노출이 됩니다.
2개 이하 일치하는 '미당첨' 타입을 추가하면 어떨까요?
class LottoRoundMachineTest { | ||
@DisplayName("로또 번호 생성 테스트, 검증된 라이브러리만 사용하여 LottoNumber 객체 생성만 확인") | ||
@Test | ||
public void lottoNumber() throws Exception { | ||
Assertions.assertThat(LottoMachine.generateLottoNumber()) | ||
.isNotNull(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
테스트의 의도가 분명하지 않은 것 같아요.
마치 통합테스트를 의도하신 것 같기도 느낌인데요
LottoRoundMachine 이라는 클래스도 없고, null 비교만으로 검증이 충분한지도 검토할 필요가 있다고 생각해요.
리뷰 주신 부분 수정했고 재리뷰 부탁드려요. LottoNumbers 클래스의 번호의 범위에 대한 책임을 분리하다보니 LottoNumbersCondition 클래스를 추가했고 LottoMachine(LottoRoundMachine 클래스는 일반 클래스 이름 바꾼걸 같이 적용 안했네요.) 도 LottoNumberCondition 클래스에서 상수 설정값을 가져오다 보니 테스트는 편해졌지만 조금 어글리한 부분이 생겼습니다.
혹시 이부분에 대해서는 다른 대안 있을까요? |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
안녕하세요 찬성님 😃
피드백 반영 잘해주셨네요 👍
질문에 대한 답변과 소소한 코멘트 남겨두었습니다.
지난 리뷰에서 놓치신 코멘트도 있는 것 같은데 확인해 주시고 리뷰 요청해 주세요.
src/main/java/lotto/LottoMain.java
Outdated
Map<LottoRank, Integer> lottoRankToCountMap = lottoRound.checkLottoRank(lottoNumbers); | ||
double rateOfReturn = (double) LottoStore.receiveWinnings(lottoRankToCountMap) / (budget * LottoStore.LOTTO_PRICE); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
수익률 계산 로직도 객체에게 메시지를 보내 구하면 어떨까요?
Map<LottoRank, Integer> lottoRankToCountMap = lottoRound.checkLottoRank(lottoNumbers); | |
double rateOfReturn = (double) LottoStore.receiveWinnings(lottoRankToCountMap) / (budget * LottoStore.LOTTO_PRICE); | |
MatchResult matchResult = lottoRound.checkLottoRank(lottoNumbers); | |
double rateOfReturn = matchResult.returnRate(구입금액); |
|
||
import java.util.Set; | ||
|
||
public class LottoNumbersCondition { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
의도가 조금 잘못 전달된 것 같아요 😅
책임을 위임한다는 것이 단순히 조건을 분리하기 보다 책임을 수행할 수 있는 객체(상태+행위)를 도출해 보시면 좋을 것 같아요.
new LottoNumber(45); // 생성 가능
new LottoNumber(46); // 생성 불가
new LottoNumber(45).equals(new LottoNumber(45)); // true
Condition 클래스를 제거하고 원시 값을 포장한 객체를 도출한다면 질문 하셨던 '매번 인스턴스를 생성하는 상황'도 해결할 수 있을 것 같아요.
src/main/java/lotto/LottoRank.java
Outdated
FOURTH(LottoNumbersCondition.getDefaultInstance().getLottoNumberSize() - 3, 5000), | ||
THIRD(LottoNumbersCondition.getDefaultInstance().getLottoNumberSize() - 2, 50000), | ||
SECOND(LottoNumbersCondition.getDefaultInstance().getLottoNumberSize() - 1, 1500000), | ||
FIRST(LottoNumbersCondition.getDefaultInstance().getLottoNumberSize(), 2000000000), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
재리뷰 부탁드려요. 입력을 재귀로 받는 것의 경우 사용자가 무한히 실패하게되면 스택 오버플로우가 발생할 것이라 우선 유지했습니다. matchCount를 수식으로 표현한 것은 로또 자체가 국가별로 숫자 및 번호 수가 다르지만 기본룰은 비슷하기 때문에 상수로 빼두고 그 값을 사용하게 했습니다. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
안녕하세요 찬성님 😃
피드백 반영 잘해주셨네요 👍
소소하게 코멘트 남겨두었는데 확인해 주시고 다음 단계 진행해 주세요!
package lotto; | ||
|
||
public class LottoNumber implements Comparable<LottoNumber> { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
불변 객체 👍
도메인 객체들은 별도의 패키지로 분리해보시면 좋을 것 같아요.
|
||
public LottoNumbers(Set<Integer> numbers) { | ||
this(LottoNumbersCondition.getDefaultInstance(), numbers); | ||
public final static int LOTTO_SIZE = 6; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
public final static int LOTTO_SIZE = 6; | |
public static final int LOTTO_SIZE = 6; |
FOURTH(LOTTO_SIZE - 3, 5000), | ||
THIRD(LOTTO_SIZE - 2, 50000), | ||
SECOND(LOTTO_SIZE - 1, 1500000), | ||
FIRST(LOTTO_SIZE, 2000000000), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
matchCount를 수식으로 표현한 것은 로또 자체가 국가별로 숫자 및 번호 수가 다르지만 기본룰은 비슷하기 때문에 상수로 빼두고 그 값을 사용하게 했습니다.
여러 상황을 고려하셨군요!
다만 당첨금은 원화 기준에다가 당첨 금액도 달라질 수 있으니 다양한 상황에도 재사용이 가능할지는 고민해 보셔도 좋을 것 같아요.
class LottoNumberTest { | ||
@DisplayName("로또 숫자 범위 테스트") |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
👍
import static org.assertj.core.api.Assertions.assertThat; | ||
|
||
class LottoResultTest { | ||
@DisplayName("로또 결과 테스트") |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
DisplayName에 총 당첨금, 수익률 계산 이라는 의도를 명시하면 어떨까요?
리뷰 부탁드려요.