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

[로또] 조천산 미션 제출합니다. #381

Open
wants to merge 8 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
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
112 changes: 112 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
@@ -1 +1,113 @@
# javascript-lotto-precourse

# 기능 요구 사항

간단한 로또 발매기를 구현한다.

- 로또 번호의 숫자 범위는 1~45까지이다.
- 1개의 로또를 발행할 때 중복되지 않는 6개의 숫자를 뽑는다.
- 당첨 번호 추첨 시 중복되지 않는 숫자 6개와 보너스 번호 1개를 뽑는다.
- 당첨은 1등부터 5등까지 있다. 당첨 기준과 금액은 아래와 같다.
- 1등: 6개 번호 일치 / 2,000,000,000원
- 2등: 5개 번호 + 보너스 번호 일치 / 30,000,000원
- 3등: 5개 번호 일치 / 1,500,000원
- 4등: 4개 번호 일치 / 50,000원
- 5등: 3개 번호 일치 / 5,000원
- 로또 구입 금액을 입력하면 구입 금액에 해당하는 만큼 로또를 발행해야 한다.
- 로또 1장의 가격은 1,000원이다.
- 당첨 번호와 보너스 번호를 입력받는다.
- 사용자가 구매한 로또 번호와 당첨 번호를 비교하여 당첨 내역 및 수익률을 출력하고 로또 게임을 종료한다.
- 사용자가 잘못된 값을 입력할 경우 "[ERROR]"로 시작하는 메시지와 함께 `Error`를 발생시키고 해당 메시지를 출력한 다음 해당 지점부터 다시 입력을 받는다.

# 프로그래밍 요구 사항

- indent(인덴트, 들여쓰기) depth를 3이 넘지 않도록 구현 2까지만 허용
- 3항 연산자를 쓰지 않는다.
- 함수(또는 메서드)가 한 가지 일만 하도록 최대한 작게 만들기
- 함수(또는 메서드)의 길이가 15라인을 넘어가지 않도록 구현
- 함수(또는 메서드)가 한 가지 일만 잘 하도록 구현
- else를 지양

구현한 기능에 대한 단위 테스트를 작성한다. 단, UI(System.out, System.in, Scanner) 로직은 제외

- 단위 테스트 작성이 익숙하지 않다면 `LottoTest`를 참고하여 학습한 후 테스트를 작성한다.

# 기능 목록

## 입력

1. 로또 구입 금액을 입력 받는다
1. 구입 금액은 1000원 단위로 입력
2. 로또 1장은 1000원
3. 1000원으로 나누어 떨어지지 않으면 [ERROR]
2. 당첨 번호를 입력 받는다
1. 번호는 쉼표(,)를 기준으로 구분
2. 중복되지 않는 숫자 6개를 입력 받는다
3. 보너스 번호를 입력 받는다
1. 보너스 번호는 1개

## 출력

1. 발행한 로또 수량 출력
2. 발행한 로또 번호 출력

```jsx
8개를 구매했습니다.
[8, 21, 23, 41, 42, 43]
[3, 5, 11, 16, 32, 38]
[7, 11, 16, 35, 36, 44]
[1, 8, 11, 31, 41, 42]
[13, 14, 16, 38, 42, 45]
[7, 11, 30, 40, 42, 43]
[2, 13, 22, 32, 38, 45]
[1, 3, 5, 14, 22, 45]
```

1. 당첨 내역 출력

```jsx
3개 일치 (5,000원) - 1개
4개 일치 (50,000원) - 0개
5개 일치 (1,500,000원) - 0개
5개 일치, 보너스 볼 일치 (30,000,000원) - 0개
6개 일치 (2,000,000,000원) - 0개
```

1. 수익률 출력
1. 수익률은 소수점 둘째 자리에서 반올림

```jsx
총 수익률은 62.5%입니다.
```

예시

```jsx
구입금액을 입력해 주세요.
8000

8개를 구매했습니다.
[8, 21, 23, 41, 42, 43]
[3, 5, 11, 16, 32, 38]
[7, 11, 16, 35, 36, 44]
[1, 8, 11, 31, 41, 42]
[13, 14, 16, 38, 42, 45]
[7, 11, 30, 40, 42, 43]
[2, 13, 22, 32, 38, 45]
[1, 3, 5, 14, 22, 45]

당첨 번호를 입력해 주세요.
1,2,3,4,5,6

보너스 번호를 입력해 주세요.
7

당첨 통계
---
3개 일치 (5,000원) - 1개
4개 일치 (50,000원) - 0개
5개 일치 (1,500,000원) - 0개
5개 일치, 보너스 볼 일치 (30,000,000원) - 0개
6개 일치 (2,000,000,000원) - 0개
총 수익률은 62.5%입니다.
```
53 changes: 50 additions & 3 deletions __tests__/LottoTest.js
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import Lotto from "../src/Lotto";
import Lotto from "../src/models/Lotto.js";

describe("로또 클래스 테스트", () => {
test("로또 번호의 개수가 6개가 넘어가면 예외가 발생한다.", () => {
Expand All @@ -7,12 +7,59 @@ describe("로또 클래스 테스트", () => {
}).toThrow("[ERROR]");
});

// TODO: 테스트가 통과하도록 프로덕션 코드 구현
test("로또 번호가 6개 미만이면 예외가 발생한다.", () => {
expect(() => {
new Lotto([1, 2, 3, 4, 5]);
}).toThrow("[ERROR]");
});

test("로또 번호에 중복된 숫자가 있으면 예외가 발생한다.", () => {
expect(() => {
new Lotto([1, 2, 3, 4, 5, 5]);
}).toThrow("[ERROR]");
});

// TODO: 추가 기능 구현에 따른 테스트 코드 작성
test.each([
[[0, 1, 2, 3, 4, 5], "1보다 작은 숫자"],
[[1, 2, 3, 4, 5, 46], "45보다 큰 숫자"],
])("로또 번호가 1~45 범위를 벗어나면 예외가 발생한다. (%s)", (numbers) => {
expect(() => {
new Lotto(numbers);
}).toThrow("[ERROR]");
});

test("로또 번호는 오름차순으로 정렬되어야 한다.", () => {
const lotto = new Lotto([6, 5, 4, 3, 2, 1]);
expect(lotto.getNumbers()).toEqual([1, 2, 3, 4, 5, 6]);
});

test("로또 인스턴스의 숫자는 변경할 수 없다.", () => {
const lotto = new Lotto([1, 2, 3, 4, 5, 6]);
const numbers = lotto.getNumbers();
numbers[0] = 7;
expect(lotto.getNumbers()).toEqual([1, 2, 3, 4, 5, 6]);
});

describe("당첨 번호 비교 테스트", () => {
test("6개 번호가 모두 일치하는 경우", () => {
const lotto = new Lotto([1, 2, 3, 4, 5, 6]);
const result = lotto.match([1, 2, 3, 4, 5, 6], 7);
expect(result.matchCount).toBe(6);
expect(result.hasBonus).toBe(false);
});

test("5개 번호와 보너스 번호가 일치하는 경우", () => {
const lotto = new Lotto([1, 2, 3, 4, 5, 7]);
const result = lotto.match([1, 2, 3, 4, 5, 6], 7);
expect(result.matchCount).toBe(5);
expect(result.hasBonus).toBe(true);
});

test("5개 번호만 일치하는 경우", () => {
const lotto = new Lotto([1, 2, 3, 4, 5, 8]);
const result = lotto.match([1, 2, 3, 4, 5, 6], 7);
expect(result.matchCount).toBe(5);
expect(result.hasBonus).toBe(false);
});
});
});
16 changes: 15 additions & 1 deletion src/App.js
Original file line number Diff line number Diff line change
@@ -1,5 +1,19 @@
import LottoGame from "./domain/LottoGame.js";

class App {
async run() {}
#lottoGame;

constructor() {
this.#lottoGame = new LottoGame();
}

async run() {
try {
await this.#lottoGame.play();
} catch (error) {
Console.print(error.message);
}
}
}

export default App;
18 changes: 0 additions & 18 deletions src/Lotto.js

This file was deleted.

8 changes: 8 additions & 0 deletions src/constants/errorMessageList.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
export const ERROR_MESSAGE = {
INVALID_AMOUNT: "[ERROR] 올바른 금액을 입력해 주세요.",
NOT_THOUSAND_UNIT: "[ERROR] 1000원 단위로 입력해 주세요.",
INVALID_NUMBER_LENGTH: "[ERROR] 로또 번호는 6개여야 합니다.",
INVALID_NUMBER_RANGE: "[ERROR] 로또 번호는 1부터 45 사이의 숫자여야 합니다.",
DUPLICATE_NUMBER: "[ERROR] 로또 번호는 중복될 수 없습니다.",
DUPLICATE_BONUS: "[ERROR] 보너스 번호는 당첨 번호와 중복될 수 없습니다.",
};
1 change: 1 addition & 0 deletions src/constants/lottoPrice.js
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
export const LOTTO_PRICE = 1000;
7 changes: 7 additions & 0 deletions src/constants/prizeList.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
export const PRIZE_List = {
THREE: 5000,
FOUR: 50000,
FIVE: 1500000,
FIVE_BONUS: 30000000,
SIX: 2000000000,
};
113 changes: 113 additions & 0 deletions src/domain/LottoGame.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,113 @@
import { Console } from "@woowacourse/mission-utils";
import { Random } from "@woowacourse/mission-utils";
import Lotto from "../models/Lotto.js";
import WinningResult from "../models/WinningResult.js";

class LottoGame {
#lottos = [];
#winningResult;
static LOTTO_PRICE = 1000;

async play() {
try {
await this.#purchaseLottos();
await this.#drawWinningNumbers();
this.#showResult();
} catch (error) {
Console.print(error.message);
}
}

async #purchaseLottos() {
const amount = await this.#getValidAmount();
const count = amount / LottoGame.LOTTO_PRICE;

this.#generateLottos(count);
this.#printPurchaseInfo(count);
}

async #getValidAmount() {
const input = await Console.readLineAsync("구입금액을 입력해 주세요.\n");
const amount = Number(input);

if (isNaN(amount) || amount <= 0) {
throw new Error("[ERROR] 올바른 금액을 입력해 주세요.");
}
if (amount % LottoGame.LOTTO_PRICE !== 0) {
throw new Error("[ERROR] 1000원 단위로 입력해 주세요.");
}

return amount;
}

#generateLottos(count) {
for (let i = 0; i < count; i++) {
const numbers = Random.pickUniqueNumbersInRange(1, 45, 6);
this.#lottos.push(new Lotto(numbers));
}
}

#printPurchaseInfo(count) {
Console.print(`\n${count}개를 구매했습니다.`);
this.#lottos.forEach((lotto) => {
Console.print(`[${lotto.getNumbers().join(", ")}]`);
});
Console.print("");
}

async #drawWinningNumbers() {
const winningNumbers = await this.#getWinningNumbers();
const bonusNumber = await this.#getBonusNumber(winningNumbers);

this.#calculateResults(winningNumbers, bonusNumber);
}

async #getWinningNumbers() {
const input = await Console.readLineAsync("당첨 번호를 입력해 주세요.\n");
const numbers = input.split(",").map((num) => Number(num.trim()));
return new Lotto(numbers).getNumbers();
}

async #getBonusNumber(winningNumbers) {
const input = await Console.readLineAsync("보너스 번호를 입력해 주세요.\n");
const number = Number(input);

if (isNaN(number) || number < 1 || number > 45) {
throw new Error("[ERROR] 보너스 번호는 1부터 45 사이의 숫자여야 합니다.");
}
if (winningNumbers.includes(number)) {
throw new Error("[ERROR] 보너스 번호는 당첨 번호와 중복될 수 없습니다.");
}

return number;
}

#calculateResults(winningNumbers, bonusNumber) {
this.#winningResult = new WinningResult();

this.#lottos.forEach((lotto) => {
const { matchCount, hasBonus } = lotto.match(winningNumbers, bonusNumber);
this.#winningResult.addResult(matchCount, hasBonus);
});
}

#showResult() {
const results = this.#winningResult.getResults();
const totalPrize = this.#winningResult.calculateTotalPrize();
const purchaseAmount = this.#lottos.length * LottoGame.LOTTO_PRICE;

Console.print("\n당첨 통계\n---");
Console.print(`3개 일치 (5,000원) - ${results[3]}개`);
Console.print(`4개 일치 (50,000원) - ${results[4]}개`);
Console.print(`5개 일치 (1,500,000원) - ${results[5]}개`);
Console.print(
`5개 일치, 보너스 볼 일치 (30,000,000원) - ${results[5.5]}개`
);
Console.print(`6개 일치 (2,000,000,000원) - ${results[6]}개`);

const profitRate = (totalPrize / purchaseAmount) * 100;
Console.print(`총 수익률은 ${profitRate.toFixed(1)}%입니다.\n`);
}
}

export default LottoGame;
Loading