Skip to content
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

[1단계 - 블랙잭 게임 실행] 커찬(이충안) 미션 제출합니다. #8

Open
wants to merge 62 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
62 commits
Select commit Hold shift + click to select a range
870ad1a
docs : 기능 명세 및 클래스 설계 작성
leegwichan Mar 5, 2024
a0a1235
feat (Card) : 카드에 따라 점수를 반환하는 기능 구현
leegwichan Mar 5, 2024
2b505f1
feat (Deck) : 카드를 한 장 뽑는 기능 구현
leegwichan Mar 5, 2024
f774cc7
feat (Dealer) : 딜러의 점수 계산 및 카드를 뽑을 수 있는지 확인하는 메서드 구현
leegwichan Mar 5, 2024
3e83ed2
style : 가독성을 위해 클래스 상단에 개행 추가, 클래스 아래 불필요한 개행 제거
leegwichan Mar 5, 2024
99baedb
feat (Dealer) : 딜러 턴 진행 구현
leegwichan Mar 6, 2024
de19b03
feat (Player) : 점수 계산 기능 구현
leegwichan Mar 6, 2024
1da0e57
feat (Player) : 딜러의 점수 계산 및 카드를 뽑을 수 있는지 확인하는 메서드 구현
leegwichan Mar 6, 2024
ed6c932
feat (Player) : 카드를 한 장 추가하는 기능 구현
leegwichan Mar 6, 2024
e11dee2
refactor (Dealer) : 요구 사항을 반영을 위해 턴 진행 기능을 카드를 한 장 추가하는 기능으로 변경
leegwichan Mar 6, 2024
877680b
refactor : 도메인 구분을 위한 participant 패키지 분리
leegwichan Mar 6, 2024
d7e7685
refactor (Participant) : 참가자의 공통 책임을 묶기 위해, 추상 클래스 사용
leegwichan Mar 6, 2024
bcf2305
docs : 이름에 대한 요구사항 추가
leegwichan Mar 6, 2024
4b8ca83
feat (Name) : 요구사항에 따른 이름 검증 기능 구현
leegwichan Mar 6, 2024
3a71ec4
feat (Player) : 플레이어가 이름을 갖도록 구현
leegwichan Mar 6, 2024
717e0cb
refactor : 참가자 관련 코드 개선
leegwichan Mar 6, 2024
940a3a7
feat (Participant) : 게임 시작 후 카드를 두 장 뽑는 기능 구현
leegwichan Mar 6, 2024
2079c7a
feat (Dealer) : 딜러가 플레이어와 승패를 확인하는 기능 구현
leegwichan Mar 6, 2024
541e6c2
feat (Player) : 플레이어가 딜러와 승패를 확인하는 기능 구현
leegwichan Mar 6, 2024
c5aab02
refactor (Deck) : 사용되지 않는 생성자 제거
leegwichan Mar 6, 2024
3bc66d8
feat (Players) : 제한된 수의 플레이어만 가지는 Players 생성
leegwichan Mar 7, 2024
da419b2
feat (Players) : 플레이어가 중복되지 않도록 검증 기능 구현
leegwichan Mar 7, 2024
fd9106d
feat (Deck) : 카드가 섞인 덱이 생성되도록 구현
leegwichan Mar 7, 2024
aea2457
feat (Players) : 모든 플레이어들이 시작 카드를 뽑는 기능 구현
leegwichan Mar 7, 2024
0fdf520
feat (BlackJackGame) : 전반적인 제어 기능 구현
leegwichan Mar 7, 2024
a6af44f
feat (InputView) : 한장의 카드를 더 받을지 요청하는 기능 구현
leegwichan Mar 7, 2024
1d288f1
feat (OutputView) : 시작 카드를 출력하는 기능 구현
leegwichan Mar 7, 2024
fcef46a
feat (OutputView) : 카드 출력 기능 구현
leegwichan Mar 7, 2024
f35da09
refactor (Participant) : 카드 출력 개수를 정하는 책임을 View로 넘기기 위해 시작 카드를 건네는 기능 제거
leegwichan Mar 7, 2024
7cbb8ca
feat (Dealer) : 모든 플레이어를 상대로 승패를 계산하는 기능 구현
leegwichan Mar 7, 2024
f3968ad
feat (OutputView) : 최종 결과 출력 기능 구현
leegwichan Mar 7, 2024
7c07079
refactor (Participant) : 카트 점수 계산 기능의 인덴트 개선
leegwichan Mar 7, 2024
dddca7c
style : 테스트의 DisplayName 변경
leegwichan Mar 7, 2024
785590d
refactor (Card) : Value, Shape 클래스 분리 및 패키지 변경
leegwichan Mar 7, 2024
bf63670
refactor (BlackJackGame) : 가독성을 위해 메서드 분리
leegwichan Mar 7, 2024
c23e7d2
refactor (OutputView) : 메서드 순서 조정
leegwichan Mar 7, 2024
116a613
refactor (CardTest) : 다양한 상황의 카드들을 픽스쳐로 추출
leegwichan Mar 7, 2024
0d2c7d5
style : 테스트 이름에서 경계값 제거
leegwichan Mar 7, 2024
7d5b4e4
refactor : 이름에 관한 픽스쳐 NameTest에 통합
leegwichan Mar 7, 2024
10bb3a8
style : 사용하지 않는 import 문 제거
leegwichan Mar 7, 2024
1fa7850
style : 의미 없는 공백 제거
leegwichan Mar 8, 2024
4f2994a
feat : 생성 단계에서 null을 입력받았을 경우, NullPointerException을 던지도록 변경
leegwichan Mar 8, 2024
b2c1c91
refactor (Participant) : 외부에서 사용하지 않는 변수의 접근 제어자 변경
leegwichan Mar 8, 2024
4c5d051
docs : README에서 구현에 관련된 내용들 삭제
leegwichan Mar 8, 2024
efc2b15
style (OutputView) : 메서드 이름을 추상적으로 변경
leegwichan Mar 10, 2024
d7c7f64
refactor (PlayerTurn) : 가독성을 위해 `BiConsumer` -> `PlayerTurn` 으로 변경
leegwichan Mar 10, 2024
4465e8e
refactor (Value) : 추후 유지보수의 용이성과 enum 인스턴스 자체가 값에 대한 책임을 가지기 위해 `maxV…
leegwichan Mar 10, 2024
cf3878f
refactor (Deck) : 패키지 변경
leegwichan Mar 10, 2024
f3a92c0
refactor : 테스트를 간결하게 작성하기 위해, Test용 생성자를 사용
leegwichan Mar 10, 2024
a1ca324
style : import 문 정리 및 TODO 제거
leegwichan Mar 10, 2024
cfe591e
feat (Hand) : 손패 계산의 책임을 분리하기 위해 Hand 구현
leegwichan Mar 10, 2024
ba33f16
feat (Hand) : 버스트 상태를 판단하는 기능 구현
leegwichan Mar 10, 2024
cf5c481
feat (Hand) : 블랙잭 판단 기능 구현
leegwichan Mar 10, 2024
6bccbbb
feat (Hand) : 카드를 추가하는 기능 구현
leegwichan Mar 10, 2024
770c1a9
feat (Hand) : 빈 손패인지 확인하는 기능 구현
leegwichan Mar 10, 2024
72369a7
refactor (Participant) : 카드와 관련된 책임들을 Hand 로 위임
leegwichan Mar 10, 2024
be2535a
feat (Dealer) : 승패 조건에 블랙잭 추가
leegwichan Mar 10, 2024
79068a6
refactor : 가독성을 위해 테스트 픽스처 분리
leegwichan Mar 10, 2024
f255e57
refactor : 도메인 규칙과 직결되는 경계값을 픽스처에서 제거
leegwichan Mar 10, 2024
4d88fee
refactor : 시작 카드가 몇 개 보여지는지의 책임을 뷰에서 도메인으로 이동
leegwichan Mar 10, 2024
b9feef4
refactor : DTO 사용을 최소화하기 위해, 원시값을 전달
leegwichan Mar 11, 2024
c0f458b
refactor (Hand) : 가독성 개선을 위해 점수 계산 기능 변경
leegwichan Mar 11, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
34 changes: 34 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -5,3 +5,37 @@
## 우아한테크코스 코드리뷰

- [온라인 코드 리뷰 과정](https://github.com/woowacourse/woowacourse-docs/blob/master/maincourse/README.md)

## 기능 요구 사항

- 블랙잭 게임은 딜러와 플레이어 중 카드의 합이 21 또는 21에 가장 가까운 숫자를 가지는 쪽이 이기는 게임이다.
- 각 카드는 점수를 가진다.
- 숫자 카드는 카드 숫자로 계산한다.
- King, Queen, Jack은 각각 10으로 계산한다.
- Ace는 1 또는 11로 계산한다.
- 플레이어는 이름을 가진다.
- 이름은 공백으로만 구성될 수 없다.
- 이름은 앞뒤 공백을 가질 수 없다.
- 중복된 이름은 가질 수 없다.

### 승리 조건

- 카드 숫자를 합쳐 가능한 21에 가깝게 만들면 이긴다.
- 처음 받은 2장 합쳐 21이 나오는 경우 블랙잭이 되며, 모든 카드조합 중에 가장 강력하다.
- 카드 숫자의 합이 21을 초과하게 되는 순간 '버스트'라고 하며 딜러의 결과에 관계없이 플레이어가 패배한다.
- 딜러와 플레이어의 점수와 블랙잭 여부가 동일한 경우, 딜러가 승리한다.

### 프로그램 진행 순서

1. 참가자 이름 입력한다.
2. 카드를 2장씩 나눠준 후에, 현재 카드 상태 출력한다.
3. 각 참가자들의 턴을 진행한다.
1. 카드가 21을 넘은 경우, 턴이 종료된다.
2. 카드를 더 받을 지 여부를 입력한다.
3. 만약 더 받는다면, 카드를 한 장 추가하고 1으로 돌아간다.
4. 더 받지 않을 경우, 턴을 종료한다.
4. 딜러의 턴을 진행한다.
1. 카드가 16을 넘은 경우, 턴이 종료된다.
2. 카드를 한 장 더 추가하고, 1로 돌아간다.
5. 딜러 및 모든 참가자의 보유 카드들과 점수를 출력한다.
6. 딜러 및 모든 참가자의 승패를 출력한다.
Empty file removed src/main/java/.gitkeep
Empty file.
8 changes: 8 additions & 0 deletions src/main/java/blackjack/BlackJackApplication.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
package blackjack;

public class BlackJackApplication {
public static void main(String[] args) {
BlackJackGame blackJackGame = new BlackJackGame();
blackJackGame.run();
}
}
60 changes: 60 additions & 0 deletions src/main/java/blackjack/BlackJackGame.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,60 @@
package blackjack;

import blackjack.domain.card.Deck;
import blackjack.domain.participant.Dealer;
import blackjack.domain.participant.Player;
import blackjack.domain.participant.Players;
import blackjack.view.InputView;
import blackjack.view.OutputView;
import java.util.List;

public class BlackJackGame {

private final InputView inputView = new InputView();
private final OutputView outputView = new OutputView();

public void run() {
Deck deck = Deck.createShuffledDeck();
Dealer dealer = new Dealer();
Players players = createPlayers();
drawStartCards(dealer, players, deck);
play(players, dealer, deck);
printResult(dealer, players);
}
Comment on lines +16 to +23
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

도메인 로직과 뷰 로직이 엉켜있는 미션이었는데 깔끔하게 처리하셨네요 👍


private Players createPlayers() {
List<String> names = inputView.inputPlayerNames();
return Players.from(names);
}

private void drawStartCards(Dealer dealer, Players players, Deck deck) {
dealer.drawStartCards(deck);
players.drawStartCards(deck);
outputView.printStartStatus(dealer, players);
}

private void play(Players players, Dealer dealer, Deck deck) {
players.play(this::playTurn, deck);
while (dealer.isDrawable()) {
outputView.printDealerDraw();
dealer.add(deck.draw());
}
Comment on lines +36 to +41
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

아이디어가 번뜩이시는군요 👍

뷰로직과 엉켜있는 플레이어 덱 완성 로직을 컨트롤러의 메서드 레퍼런스로 해결하셨군요
배우고 갑니다 💪🏻

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

저는 정말 이 방법이 좋은 방법인지 고민되긴 해요... 제가 받았던 리뷰에는 이런 내용도 있었어요!

코드상으로는 의존성이 분리된 것 처럼 보이지만, 런타임에 도메인 객체 내에서 BiConsumer로 View의 영향을 받는 로직이 주입되어 수행될 수도 있겠어요.
BlackJackGame.playTurn 메서드에 핵심적인 도메인 로직이 노출되어 있는 것 같기도 하네요.
view에서 값을 받아온 후에 넘겨주어 이곳에 핵심 도메인 규칙들이 위치하도록 변경해보는 것은 어떤가요?

}

private void playTurn(Player player, Deck deck) {
while (player.isDrawable() && inputView.isPlayerWantDraw(player.getName())) {
player.add(deck.draw());
outputView.printPlayerCards(player);
}
}

private void printResult(Dealer dealer, Players players) {
outputView.printEndingStatus(dealer, players);
int winCount = dealer.calculateWinCount(players);
int loseCount = dealer.calculateLoseCount(players);
outputView.printDealerMatchResult(winCount, loseCount);
for (Player player : players.getPlayers()) {
outputView.printPlayerMatchResult(player.getName(), player.isWin(dealer));
}
}
}
51 changes: 51 additions & 0 deletions src/main/java/blackjack/domain/card/Card.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
package blackjack.domain.card;

import java.util.Objects;

public class Card {

private final Value value;
private final Shape shape;

public Card(Value value, Shape shape) {
this.value = Objects.requireNonNull(value);
this.shape = Objects.requireNonNull(shape);
Comment on lines +11 to +12
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

null 검증 👍

}

public int getMinScore() {
return value.getMinScore();
}

public int getMaxScore() {
return value.getMaxScore();
}

@Override
public boolean equals(Object object) {
if (this == object) {
return true;
}
if (object == null || getClass() != object.getClass()) {
return false;
}
Card card = (Card) object;
return value == card.value && shape == card.shape;
}

@Override
public int hashCode() {
return Objects.hash(value, shape);
}

public Value getValue() {
return value;
}

public Shape getShape() {
return shape;
}

public boolean isAce() {
return value.isAce();
}
}
40 changes: 40 additions & 0 deletions src/main/java/blackjack/domain/card/Deck.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
package blackjack.domain.card;

import static java.util.stream.Collectors.toList;

import java.util.Arrays;
import java.util.Collections;
import java.util.LinkedList;
import java.util.List;
import java.util.Queue;

public class Deck {

private final Queue<Card> cards;

private Deck(List<Card> cards) {
this.cards = new LinkedList<>(cards);
}

public static Deck createShuffledDeck() {
List<Card> cards = Arrays.stream(Shape.values())
.map(Deck::makeCards)
.flatMap(List::stream)
.collect(toList());
Collections.shuffle(cards);
return new Deck(cards);
}

private static List<Card> makeCards(Shape shape) {
return Arrays.stream(Value.values())
.map(value -> new Card(value, shape))
.toList();
}

public Card draw() {
if (cards.isEmpty()) {
throw new IllegalStateException("카드를 더 이상 뽑을 수 없습니다.");
}
return cards.poll();
}
}
67 changes: 67 additions & 0 deletions src/main/java/blackjack/domain/card/Hand.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,67 @@
package blackjack.domain.card;

import java.util.ArrayList;
import java.util.Collections;
import java.util.List;

public class Hand {

private static final int BLACKJACK_SCORE = 21;
private static final int BLACKJACK_SIZE = 2;

private final List<Card> cards;

public Hand(List<Card> cards) {
this.cards = new ArrayList<>(cards);
}

public void add(Card card) {
cards.add(card);
}

public int calculateScore() {
int totalMinScore = getMinScore();
int biggerScore = getBiggerScore();

if (biggerScore > BLACKJACK_SCORE) {
return totalMinScore;
}
return biggerScore;
}

private int getMinScore() {
return cards.stream()
.mapToInt(Card::getMinScore)
.sum();
}

private int getBiggerScore() {
int score = getMinScore();
int differenceScore = cards.stream()
.filter(Card::isAce)
.mapToInt(this::calculateDifferenceScore)
.findAny()
.orElse(0);
return score + differenceScore;
}

private int calculateDifferenceScore(Card card) {
return card.getMaxScore() - card.getMinScore();
}

public boolean isBusted() {
return calculateScore() > BLACKJACK_SCORE;
}

public boolean isBlackjack() {
return cards.size() == BLACKJACK_SIZE && calculateScore() == BLACKJACK_SCORE;
}

public List<Card> getCards() {
return Collections.unmodifiableList(cards);
}

public boolean isEmpty() {
return cards.isEmpty();
}
}
5 changes: 5 additions & 0 deletions src/main/java/blackjack/domain/card/Shape.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
package blackjack.domain.card;

public enum Shape {
SPADE, DIAMOND, HEART, CLOVER
}
37 changes: 37 additions & 0 deletions src/main/java/blackjack/domain/card/Value.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
package blackjack.domain.card;

public enum Value {
ACE(1, 11),
TWO(2, 2),
THREE(3, 3),
FOUR(4, 4),
FIVE(5, 5),
SIX(6, 6),
SEVEN(7, 7),
EIGHT(8, 8),
NINE(9, 9),
TEN(10, 10),
JACK(10, 10),
QUEEN(10, 10),
KING(10, 10);

private final int minScore;
private final int maxScore;

Value(int minScore, int maxScore) {
this.minScore = minScore;
this.maxScore = maxScore;
}

public int getMinScore() {
return minScore;
}

public int getMaxScore() {
return maxScore;
}

public boolean isAce() {
return this == ACE;
}
}
55 changes: 55 additions & 0 deletions src/main/java/blackjack/domain/participant/Dealer.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,55 @@
package blackjack.domain.participant;

import blackjack.domain.card.Card;
import java.util.Collections;
import java.util.List;

public class Dealer extends Participant {

Comment on lines +6 to +8
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

상속에 대한 커찬의 생각 궁금합니다. 개인적으로는 상속에 여러 단점을 높게 평가하는 편인데요

한번 읽어보셔도 좋을 만한 참고글 첨부합니다 👍
https://tecoble.techcourse.co.kr/post/2020-05-18-inheritance-vs-composition/

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

상속보다는 합성을 사용하라는 말, 저도 많이 들었던 것 같아요. 저는 "지금 코드에서 Participant에 따라 DealerPlayer가 크리티컬하게 문제가 생길까?에 대해 고민해보고, 문제가 없어 사용한 것 같아요.

상속의 단점들을 보완하기 위해. 자체적인 규칙들을 도입해봤어요~

  • 상속을 이용할 때는 '왠만하면' 추상 클래스를 이용한다.
    • 직접 객체가 되면서 다른 클래스의 상위 클래스가 된다고 하면, 너무 많은 역할을 가지게 되어 유지보수가 힘들 것 같았어요.
  • 추상 클래스의 모든 메서드는 final 또는 abstract로 선언한다.
    • 외부로 제공하는 public method와 서로 다른 부분을 나타내는 abstract method를 구분해서 공용 메서드를 오버라이딩을 막아 최대한 예측되지 않는 일이 일어나는 것을 방지했어요.

private static final int DRAWABLE_MAX_SCORE = 16;
private static final int START_CARD_SIZE = 1;

public Dealer() {
super(Collections.emptyList());
}

Dealer(List<Card> cards) {
super(cards);
}

public boolean isWin(Player player) {
if (player.isBusted() || this.isBlackjack()) {
return true;
}
if (this.isBusted() || player.isBlackjack()) {
return false;
}
return this.calculateScore() >= player.calculateScore();
}

private boolean isLose(Player player) {
return !isWin(player);
}

public int calculateWinCount(Players players) {
return (int) players.getPlayers().stream()
.filter(this::isWin)
.count();
}

public int calculateLoseCount(Players players) {
return (int) players.getPlayers().stream()
.filter(this::isLose)
.count();
}

@Override
protected int getMaxDrawableScore() {
return DRAWABLE_MAX_SCORE;
}

@Override
protected int getStartCardSize() {
return START_CARD_SIZE;
}
}
Loading