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

๐Ÿš€ 2๋‹จ๊ณ„ - ๋ธ”๋ž™์žญ #570

Open
wants to merge 11 commits into
base: rladowl92
Choose a base branch
from
Open
Changes from all commits
Commits
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
39 changes: 38 additions & 1 deletion README.md
Original file line number Diff line number Diff line change
@@ -1 +1,38 @@
# kotlin-blackjack
# kotlin-blackjack
### ๊ธฐ๋Šฅ ์š”๊ตฌ์‚ฌํ•ญ
- ๋ธ”๋ž™์žญ ๊ฒŒ์ž„์„ ๋ณ€ํ˜•ํ•œ ํ”„๋กœ๊ทธ๋žจ์„ ๊ตฌํ˜„ํ•œ๋‹ค. ๋ธ”๋ž™์žญ ๊ฒŒ์ž„์€ ๋”œ๋Ÿฌ์™€ ํ”Œ๋ ˆ์ด์–ด ์ค‘ ์นด๋“œ์˜ ํ•ฉ์ด 21 ๋˜๋Š” 21์— ๊ฐ€์žฅ ๊ฐ€๊นŒ์šด ์ˆซ์ž๋ฅผ ๊ฐ€์ง€๋Š” ์ชฝ์ด ์ด๊ธฐ๋Š” ๊ฒŒ์ž„์ด๋‹ค.

- ์นด๋“œ์˜ ์ˆซ์ž ๊ณ„์‚ฐ์€ ์นด๋“œ ์ˆซ์ž๋ฅผ ๊ธฐ๋ณธ์œผ๋กœ ํ•˜๋ฉฐ, ์˜ˆ์™ธ๋กœ Ace๋Š” 1 ๋˜๋Š” 11๋กœ ๊ณ„์‚ฐํ•  ์ˆ˜ ์žˆ์œผ๋ฉฐ, King, Queen, Jack์€ ๊ฐ๊ฐ 10์œผ๋กœ ๊ณ„์‚ฐํ•œ๋‹ค.
- ๊ฒŒ์ž„์„ ์‹œ์ž‘ํ•˜๋ฉด ํ”Œ๋ ˆ์ด์–ด๋Š” ๋‘ ์žฅ์˜ ์นด๋“œ๋ฅผ ์ง€๊ธ‰ ๋ฐ›์œผ๋ฉฐ, ๋‘ ์žฅ์˜ ์นด๋“œ ์ˆซ์ž๋ฅผ ํ•ฉ์ณ 21์„ ์ดˆ๊ณผํ•˜์ง€ ์•Š์œผ๋ฉด์„œ 21์— ๊ฐ€๊น๊ฒŒ ๋งŒ๋“ค๋ฉด ์ด๊ธด๋‹ค. 21์„ ๋„˜์ง€ ์•Š์„ ๊ฒฝ์šฐ ์›ํ•œ๋‹ค๋ฉด ์–ผ๋งˆ๋“ ์ง€ ์นด๋“œ๋ฅผ ๊ณ„์† ๋ฝ‘์„ ์ˆ˜ ์žˆ๋‹ค.

### ์‹คํ–‰ ๊ฒฐ๊ณผ
```
๊ฒŒ์ž„์— ์ฐธ์—ฌํ•  ์‚ฌ๋žŒ์˜ ์ด๋ฆ„์„ ์ž…๋ ฅํ•˜์„ธ์š”.(์‰ผํ‘œ ๊ธฐ์ค€์œผ๋กœ ๋ถ„๋ฆฌ)
pobi,jason

pobi, jason์—๊ฒŒ 2์žฅ์˜ ์นด๋“œ๋ฅผ ๋‚˜๋ˆ„์—ˆ์Šต๋‹ˆ๋‹ค.
pobi์นด๋“œ: 2ํ•˜ํŠธ, 8์ŠคํŽ˜์ด๋“œ
jason์นด๋“œ: 7ํด๋กœ๋ฒ„, K์ŠคํŽ˜์ด๋“œ

pobi๋Š” ํ•œ์žฅ์˜ ์นด๋“œ๋ฅผ ๋” ๋ฐ›๊ฒ ์Šต๋‹ˆ๊นŒ?(์˜ˆ๋Š” y, ์•„๋‹ˆ์˜ค๋Š” n)
y
pobi์นด๋“œ: 2ํ•˜ํŠธ, 8์ŠคํŽ˜์ด๋“œ, Aํด๋กœ๋ฒ„
pobi๋Š” ํ•œ์žฅ์˜ ์นด๋“œ๋ฅผ ๋” ๋ฐ›๊ฒ ์Šต๋‹ˆ๊นŒ?(์˜ˆ๋Š” y, ์•„๋‹ˆ์˜ค๋Š” n)
n
jason์€ ํ•œ์žฅ์˜ ์นด๋“œ๋ฅผ ๋” ๋ฐ›๊ฒ ์Šต๋‹ˆ๊นŒ?(์˜ˆ๋Š” y, ์•„๋‹ˆ์˜ค๋Š” n)
n
jason์นด๋“œ: 7ํด๋กœ๋ฒ„, K์ŠคํŽ˜์ด๋“œ

pobi์นด๋“œ: 2ํ•˜ํŠธ, 8์ŠคํŽ˜์ด๋“œ, Aํด๋กœ๋ฒ„ - ๊ฒฐ๊ณผ: 21
jason์นด๋“œ: 7ํด๋กœ๋ฒ„, K์ŠคํŽ˜์ด๋“œ - ๊ฒฐ๊ณผ: 17
```

### ๊ธฐ๋Šฅ๋ชฉ๋ก
- card
- card shape, card number
- player
- name, card list
- cardSum()
- players
- player list
- ์ž…์ถœ๋ ฅ
2 changes: 1 addition & 1 deletion build.gradle.kts
Original file line number Diff line number Diff line change
@@ -13,7 +13,7 @@ repositories {
dependencies {
testImplementation("org.junit.jupiter", "junit-jupiter", "5.8.2")
testImplementation("org.assertj", "assertj-core", "3.22.0")
testImplementation("io.kotest", "kotest-runner-junit5", "5.2.3")
testImplementation("io.kotest", "kotest-runner-junit5", "5.5.0")
}

tasks {
26 changes: 26 additions & 0 deletions src/main/kotlin/blackjack/BlackjackApplication.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
package blackjack

import blackjack.domain.Players
import blackjack.view.InputConsoleView
import blackjack.view.OutputConsoleView

object BlackjackApplication {
private val inputConsoleView = InputConsoleView()
private val outputConsoleView = OutputConsoleView()

@JvmStatic
fun main(args: Array<String>) {
val namesOfPlayers = inputConsoleView.namesOfPlayers()
val players = Players(namesOfPlayers)

outputConsoleView.printInitCardMsg(players)
players.players.map { player ->
while (player.cardSum() < 21 && inputConsoleView.wannaGetNextCard(player)) {
players.getCard(player)
outputConsoleView.printCards(player)
}
}

outputConsoleView.printResult(players)
}
}
3 changes: 3 additions & 0 deletions src/main/kotlin/blackjack/domain/Card.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
package blackjack.domain

data class Card(val shape: CardShape, val number: CardNumber)
16 changes: 16 additions & 0 deletions src/main/kotlin/blackjack/domain/CardNumber.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
package blackjack.domain

enum class CardNumber(val value: Number, val display: String) {
ACE(1, "A"),
TWO(2, "2"),
THREE(3, "3"),
FOUR(4, "4"),
FIVE(5, "5"),
SIX(6, "6"),
SEVEN(7, "7"),
EIGHT(8, "8"),
NINE(9, "9"),
JACK(10, "J"),
QUEEN(10, "Q"),
KING(10, "K"),
}
8 changes: 8 additions & 0 deletions src/main/kotlin/blackjack/domain/CardShape.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
package blackjack.domain

enum class CardShape(val display: String) {
SPADE("์ŠคํŽ˜์ด๋“œ"),
DIAMOND("๋‹ค์ด์•„"),
HEART("ํ•˜ํŠธ"),
CLOVA("ํด๋กœ๋ฒ„"),
}
17 changes: 17 additions & 0 deletions src/main/kotlin/blackjack/domain/Player.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
package blackjack.domain

class Player(val name: String, initCards: List<Card>) {
val cards: MutableList<Card> = initCards.toMutableList()

fun addCard(card: Card) {
cards.add(card)
}

fun cardSum(): Int {
val sum = cards.sumOf { it.number.value.toInt() }
if (cards.map { it.number }.contains(CardNumber.ACE) && sum <= 11) {
return sum + 10
}
return sum
}
}
33 changes: 33 additions & 0 deletions src/main/kotlin/blackjack/domain/Players.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
package blackjack.domain

class Players(names: List<String>) {
private val cards = mutableListOf<Card>()
val players = names.map { Player(it, arrayListOf(randomCard(), randomCard())) }

init {
require(names.size == names.toSet().size)
}

fun getPlayer(name: String): Player {
return players.find { it.name == name } ?: throw IllegalArgumentException("no such player")
}

fun getCard(player: Player) {
player.addCard(randomCard())
}

private fun randomCard(): Card {
val cardShape = CardShape.values().toList().shuffled()[0]
val cardNumber = CardNumber.values().toList().shuffled()[0]
val card = Card(cardShape, cardNumber)
if (hasCard(card)) {
return randomCard()
}
cards.add(card)
return card
}

private fun hasCard(card: Card): Boolean {
return cards.contains(card)
}
}
15 changes: 15 additions & 0 deletions src/main/kotlin/blackjack/view/InputConsoleView.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
package blackjack.view

import blackjack.domain.Player

class InputConsoleView {
fun namesOfPlayers(): List<String> {
println("๊ฒŒ์ž„์— ์ฐธ์—ฌํ•  ์‚ฌ๋žŒ์˜ ์ด๋ฆ„์„ ์ž…๋ ฅํ•˜์„ธ์š”.(์‰ผํ‘œ ๊ธฐ์ค€์œผ๋กœ ๋ถ„๋ฆฌ)")
return readln().split(",")
}

fun wannaGetNextCard(player: Player): Boolean {
println("${player.name}๋Š” ํ•œ์žฅ์˜ ์นด๋“œ๋ฅผ ๋” ๋ฐ›๊ฒ ์Šต๋‹ˆ๊นŒ?(์˜ˆ๋Š” y, ์•„๋‹ˆ์˜ค๋Š” n)")
return readln() == "y"
}
}
30 changes: 30 additions & 0 deletions src/main/kotlin/blackjack/view/OutputConsoleView.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
package blackjack.view

import blackjack.domain.Card
import blackjack.domain.Player
import blackjack.domain.Players

class OutputConsoleView {
fun printInitCardMsg(players: Players) {
println("${players.players.map { it.name }.joinToString(",")} ์—๊ฒŒ 2์žฅ์˜ ์นด๋“œ๋ฅผ ๋‚˜๋ˆ„์—ˆ์Šต๋‹ˆ๋‹ค.")
players.players.map { printCards(it) }
println()
}

fun printResult(players: Players) {
println()
players.players.map { println("${cardsToString(it)} - ๊ฒฐ๊ณผ: ${it.cardSum()}") }
}

fun printCards(player: Player) {
println(cardsToString(player))
}

private fun cardsToString(player: Player): String {
return "${player.name}์นด๋“œ: ${player.cards.joinToString(", ") { cardToString(it) }}"
}

private fun cardToString(card: Card): String {
return "${card.number.display}${card.shape.display}"
}
}
6 changes: 5 additions & 1 deletion src/main/kotlin/learning/LanguageBuilder.kt
Original file line number Diff line number Diff line change
@@ -1,9 +1,13 @@
package learning

class LanguageBuilder {
val languages = mutableListOf<Language>()
private val languages = mutableListOf<Language>()

infix fun String.level(level: Int) {
languages.add(Language(this, level))
}

fun build(): MutableList<Language> {
return languages
}
}
10 changes: 4 additions & 6 deletions src/main/kotlin/learning/PersonBuilder.kt
Original file line number Diff line number Diff line change
@@ -3,8 +3,8 @@ package learning
class PersonBuilder {
private lateinit var name: String
private var company: String? = null
private val skills = mutableListOf<Skill>()
private val languages = mutableListOf<Language>()
private var skills = mutableListOf<Skill>()
private var languages = mutableListOf<Language>()

fun name(value: String) {
name = value
@@ -15,13 +15,11 @@ class PersonBuilder {
}

fun skills(block: SkillsBuilder.() -> Unit) {
val skillsBuilder = SkillsBuilder().apply(block)
skills.addAll(skillsBuilder.skills)
this.skills = SkillsBuilder().apply(block).build()
}

fun languages(block: LanguageBuilder.() -> Unit) {
val languageBuilder = LanguageBuilder().apply(block)
languages.addAll(languageBuilder.languages)
this.languages = LanguageBuilder().apply(block).build()
}

fun build(): Person {
6 changes: 5 additions & 1 deletion src/main/kotlin/learning/SkillsBuilder.kt
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
package learning

class SkillsBuilder {
val skills = mutableListOf<Skill>()
private val skills = mutableListOf<Skill>()

fun soft(value: String) {
skills.add(Skill(SkillType.SOFT, value))
@@ -10,4 +10,8 @@ class SkillsBuilder {
fun hard(value: String) {
skills.add(Skill(SkillType.HARD, value))
}

fun build(): MutableList<Skill> {
return skills
}
}
10 changes: 10 additions & 0 deletions src/test/kotlin/blackjack/domain/CardTest.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
package blackjack.domain

import io.kotest.core.spec.style.StringSpec
import io.kotest.matchers.shouldBe

class CardTest : StringSpec({
"Card ์ƒ์„ฑ ์„ฑ๊ณต" {
Card(CardShape.CLOVA, CardNumber.JACK) shouldBe Card(CardShape.CLOVA, CardNumber.JACK)
}
})
48 changes: 48 additions & 0 deletions src/test/kotlin/blackjack/domain/PlayerTest.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
package blackjack.domain

import io.kotest.core.spec.style.StringSpec
import io.kotest.matchers.shouldBe

class PlayerTest : StringSpec({
val initCards = listOf(Card(CardShape.CLOVA, CardNumber.TWO), Card(CardShape.HEART, CardNumber.THREE))
var edge = Player("edge", initCards)

beforeEach {
edge = Player("edge", initCards)
}

"player ์ƒ์„ฑ ์„ฑ๊ณต" {
edge.name shouldBe "edge"
}

"player ์นด๋“œ ํ™•์ธํ•˜๊ธฐ" {
edge.cards shouldBe initCards
}

"player ์นด๋“œ ํ•ฉ์‚ฐํ•˜๊ธฐ" {
edge.cardSum() shouldBe 5
}

"player์— ์นด๋“œ ์ถ”๊ฐ€ํ•˜๊ธฐ" {
val addCard = Card(CardShape.DIAMOND, CardNumber.JACK)
edge.addCard(addCard)

edge.cardSum() shouldBe 15
val list = initCards.toMutableList()
list.add(addCard)
edge.cards shouldBe list
}

"ํ•ฉ์ด 11๋ณด๋‹ค ์ž‘๊ฑฐ๋‚˜ ๊ฐ™์œผ๋ฉด ace๋ฅผ 11๋กœ ๊ณ„์‚ฐํ•˜๊ธฐ" {
edge.addCard(Card(CardShape.DIAMOND, CardNumber.ACE))

edge.cardSum() shouldBe 16
}

"ํ•ฉ์ด 11๋ณด๋‹ค ํฌ๋ฉด ace๋ฅผ 1๋กœ ๊ณ„์‚ฐํ•˜๊ธฐ" {
edge.addCard(Card(CardShape.DIAMOND, CardNumber.ACE))
edge.addCard(Card(CardShape.DIAMOND, CardNumber.JACK))

edge.cardSum() shouldBe 16
}
})
18 changes: 18 additions & 0 deletions src/test/kotlin/blackjack/domain/PlayersTest.kt
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
package blackjack.domain

import io.kotest.assertions.throwables.shouldThrow
import io.kotest.core.spec.style.StringSpec
import io.kotest.matchers.shouldBe

class PlayersTest : StringSpec({
"players ์ƒ์„ฑ ์„ฑ๊ณต" {
val players = Players(listOf("edge", "test"))
players.getPlayer("edge").name shouldBe "edge"
}

"player ์ด๋ฆ„์ด ์ค‘๋ณต๋˜๋ฉด ์—๋Ÿฌ" {
shouldThrow<IllegalArgumentException> {
Players(listOf("edge", "edge"))
}
}
})
19 changes: 10 additions & 9 deletions src/test/kotlin/learning/DslTest.kt
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
package learning

import io.kotest.matchers.collections.shouldBeIn
import io.kotest.matchers.nulls.shouldBeNull
import io.kotest.matchers.shouldBe
import org.junit.jupiter.api.Test
@@ -53,14 +54,11 @@ class DslTest {
soft("Good communication skills")
hard("Kotlin")
}
languages {
"Korean" level 5
"English" level 3
}
}

person.languages.languages[0] shouldBe Language("Korean", 5)
person.languages.languages[1] shouldBe Language("English", 3)
Skill(SkillType.SOFT, "A passion for problem solving").shouldBeIn(person.skills.skills)
Skill(SkillType.SOFT, "Good communication skills").shouldBeIn(person.skills.skills)
Skill(SkillType.HARD, "Kotlin").shouldBeIn(person.skills.skills)
}

@Test
@@ -73,11 +71,14 @@ class DslTest {
soft("Good communication skills")
hard("Kotlin")
}
languages {
"Korean" level 5
"English" level 3
}
}

person.skills.skills[0] shouldBe Skill(SkillType.SOFT, "A passion for problem solving")
person.skills.skills[1] shouldBe Skill(SkillType.SOFT, "Good communication skills")
person.skills.skills[2] shouldBe Skill(SkillType.HARD, "Kotlin")
Language("Korean", 5).shouldBeIn(person.languages.languages)
Language("English", 3).shouldBeIn(person.languages.languages)
}
}