From 12c885230877acfbb8f2ec1ba7627822403f9c41 Mon Sep 17 00:00:00 2001
From: Taek Ki Min <127868094+TaekkiMin@users.noreply.github.com>
Date: Mon, 6 Nov 2023 00:28:19 +0900
Subject: [PATCH 1/8] =?UTF-8?q?feat=20:=20=EC=A0=95=EA=B8=B0=EC=A0=81?=
=?UTF-8?q?=EC=9C=BC=EB=A1=9C=20=EC=B6=9C=EB=A0=A5=EB=90=98=EB=8A=94=20?=
=?UTF-8?q?=EB=AC=B8=EA=B5=AC=20=EA=B5=AC=ED=98=84=ED=98=84?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
src/Constants.js | 9 +++++++++
src/Lotto.js | 1 +
2 files changed, 10 insertions(+)
create mode 100644 src/Constants.js
diff --git a/src/Constants.js b/src/Constants.js
new file mode 100644
index 0000000000..0418201e7f
--- /dev/null
+++ b/src/Constants.js
@@ -0,0 +1,9 @@
+// 정기적으로 출력되는 문구를 변수로 저장
+const GAME_MESSAGE = {
+ purchaseAmount: "구입 금액을 입력하세요\n",
+ guessNumbers: "당첨 번호를 입력하세요\n",
+ bonusNumber: "보너스 번호를 입력하세요\n",
+ static: "당첨 통계\n---",
+};
+
+module.exports = { GAME_MESSAGE };
diff --git a/src/Lotto.js b/src/Lotto.js
index cb0b1527e9..3a8fb7b7be 100644
--- a/src/Lotto.js
+++ b/src/Lotto.js
@@ -1,3 +1,4 @@
+// 입력한 구매 금액에 맞게 로또 인스터스를 생성하는 코드
class Lotto {
#numbers;
From 8d0a5ef086ac7617d7e76c0872c13bd7fbf3202b Mon Sep 17 00:00:00 2001
From: Taek Ki Min <127868094+TaekkiMin@users.noreply.github.com>
Date: Wed, 8 Nov 2023 23:37:34 +0900
Subject: [PATCH 2/8] =?UTF-8?q?constant.js=20=ED=8C=8C=EC=9D=BC=20?=
=?UTF-8?q?=EA=B5=AC=ED=98=84?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
src/Constants.js | 39 ++++++++++++++++++++++++++++++---------
1 file changed, 30 insertions(+), 9 deletions(-)
diff --git a/src/Constants.js b/src/Constants.js
index 0418201e7f..11d9383c21 100644
--- a/src/Constants.js
+++ b/src/Constants.js
@@ -1,9 +1,30 @@
-// 정기적으로 출력되는 문구를 변수로 저장
-const GAME_MESSAGE = {
- purchaseAmount: "구입 금액을 입력하세요\n",
- guessNumbers: "당첨 번호를 입력하세요\n",
- bonusNumber: "보너스 번호를 입력하세요\n",
- static: "당첨 통계\n---",
-};
-
-module.exports = { GAME_MESSAGE };
+export const ERROR_INVALID_NUMBER =
+ "[ERROR] 구매 금액은 유효한 숫자여야 합니다.";
+export const ERROR_NOT_IN_1000_UNITS =
+ "[ERROR] 구매 금액은 1,000원 단위이어야 합니다.";
+export const ERROR_INVALID_LOTTO_NUMBER_COUNT =
+ "[ERROR] 로또 번호는 6개여야 합니다.";
+export const ERROR_DUPLICATE_LOTTO_NUMBERS =
+ "[ERROR] 로또 번호는 중복되지 않은 숫자로 이루어져야 합니다.";
+
+export const LOTTO_PURCHASE_MESSAGE = "구입금액을 입력해 주세요.\n";
+export const WINNING_NUMBER_INPUT_MESSAGE = "당첨 번호를 입력해 주세요.\n";
+export const BONUS_NUMBER_INPUT_MESSAGE = "보너스 번호를 입력해 주세요.\n";
+
+export const LOTTO_STATISTICS_HEADER = "당첨 통계\n---";
+export const PRIZE_3_MATCH = "3개 일치 (5,000원) - ";
+export const PRIZE_4_MATCH = "4개 일치 (50,000원) - ";
+export const PRIZE_5_MATCH = "5개 일치 (1,500,000원) - ";
+export const PRIZE_5_MATCH_WITH_BONUS =
+ "5개 일치, 보너스 볼 일치 (30,000,000원) - ";
+export const PRIZE_6_MATCH = "6개 일치 (2,000,000,000원) - ";
+export const TOTAL_PROFIT_PERCENTAGE = "총 수익률은 ";
+
+export const LOTTO_TICKET_PRICE = 1000;
+export const LOTTO_NUMBERS_COUNT = 6;
+
+export const MATCH_3 = 5_000;
+export const MATCH_4 = 50_000;
+export const MATCH_5 = 1_500_000;
+export const MATCH_5_WITH_BONUS = 30_000_000;
+export const MATCH_6 = 2_000_000_000;
From cd8cd025417515709c4dadd0d3800c662c017f72 Mon Sep 17 00:00:00 2001
From: Taek Ki Min <127868094+TaekkiMin@users.noreply.github.com>
Date: Wed, 8 Nov 2023 23:38:58 +0900
Subject: [PATCH 3/8] =?UTF-8?q?Lotto.js=20=ED=8C=8C=EC=9D=BC=20=EA=B5=AC?=
=?UTF-8?q?=ED=98=84?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
src/Lotto.js | 29 +++++++++++++++++++++++++----
1 file changed, 25 insertions(+), 4 deletions(-)
diff --git a/src/Lotto.js b/src/Lotto.js
index 3a8fb7b7be..2ee97f8b61 100644
--- a/src/Lotto.js
+++ b/src/Lotto.js
@@ -1,19 +1,40 @@
-// 입력한 구매 금액에 맞게 로또 인스터스를 생성하는 코드
+import { Console } from "@woowacourse/mission-utils";
+
+import * as CONSTANTS from "./Constants.js";
+
class Lotto {
#numbers;
constructor(numbers) {
this.#validate(numbers);
- this.#numbers = numbers;
+ this.#numbers = numbers.sort((a, b) => a - b);
+ this.#printNumbers();
}
#validate(numbers) {
if (numbers.length !== 6) {
- throw new Error("[ERROR] 로또 번호는 6개여야 합니다.");
+ throw new Error(CONSTANTS.ERROR_INVALID_LOTTO_NUMBER_COUNT);
+ }
+ const uniqueNumbers = new Set(numbers);
+ if (uniqueNumbers.size !== 6) {
+ throw new Error(CONSTANTS.ERROR_DUPLICATE_LOTTO_NUMBERS);
}
}
- // TODO: 추가 기능 구현
+ #printNumbers() {
+ Console.print(`[${this.#numbers.join(", ")}]`);
+ }
+
+ result(winningNums, bonusNum) {
+ const res = { cnt: 0, hasBonus: false };
+ for (let i = 0; i < CONSTANTS.LOTTO_NUMBERS_COUNT; i++) {
+ if (this.#numbers.includes(winningNums[i])) {
+ res.cnt++;
+ }
+ }
+ if (this.#numbers.includes(bonusNum)) res.hasBonus = true;
+ return res;
+ }
}
export default Lotto;
From e221c5823ab8dfab7ce8b379b05ef21e66493ba7 Mon Sep 17 00:00:00 2001
From: Taek Ki Min <127868094+TaekkiMin@users.noreply.github.com>
Date: Wed, 8 Nov 2023 23:42:44 +0900
Subject: [PATCH 4/8] =?UTF-8?q?LottoUtils.js=20=ED=8C=8C=EC=9D=BC=20?=
=?UTF-8?q?=EA=B5=AC=ED=98=84?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
src/LottoUtils.js | 0
1 file changed, 0 insertions(+), 0 deletions(-)
create mode 100644 src/LottoUtils.js
diff --git a/src/LottoUtils.js b/src/LottoUtils.js
new file mode 100644
index 0000000000..e69de29bb2
From d7cd82ee5ab7c2f5f5b20dd27b7a79f41f547f62 Mon Sep 17 00:00:00 2001
From: Taek Ki Min <127868094+TaekkiMin@users.noreply.github.com>
Date: Wed, 8 Nov 2023 23:43:49 +0900
Subject: [PATCH 5/8] =?UTF-8?q?LottoUtils.js=20=ED=8C=8C=EC=9D=BC=20?=
=?UTF-8?q?=EC=9E=AC=EA=B5=AC=EA=B5=AC=ED=98=84?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
src/LottoUtils.js | 113 ++++++++++++++++++++++++++++++++++++++++++++++
1 file changed, 113 insertions(+)
diff --git a/src/LottoUtils.js b/src/LottoUtils.js
index e69de29bb2..41c60e56be 100644
--- a/src/LottoUtils.js
+++ b/src/LottoUtils.js
@@ -0,0 +1,113 @@
+import { Console, Random } from "@woowacourse/mission-utils";
+import Lotto from "./Lotto.js";
+import * as CONSTANTS from "./Constants.js";
+
+export async function getValidPurchaseAmount() {
+ let validPurchaseAmount = false;
+ let purchaseAmountInput;
+
+ while (!validPurchaseAmount) {
+ try {
+ purchaseAmountInput = await Console.readLineAsync(
+ CONSTANTS.LOTTO_PURCHASE_MESSAGE
+ );
+ if (isNaN(Number(purchaseAmountInput))) {
+ throw new Error(CONSTANTS.ERROR_INVALID_NUMBER);
+ }
+
+ if (Number(purchaseAmountInput) % CONSTANTS.LOTTO_TICKET_PRICE !== 0) {
+ throw new Error(CONSTANTS.ERROR_NOT_IN_1000_UNITS);
+ }
+
+ validPurchaseAmount = true;
+ } catch (error) {
+ Console.print(error.message);
+ }
+ }
+
+ return Number(purchaseAmountInput);
+}
+
+export function purchaseLottos(count) {
+ const lottos = [];
+ Console.print(`${count}개를 구매했습니다.`);
+ for (let i = 0; i < count; i++) {
+ const numbers = Random.pickUniqueNumbersInRange(
+ 1,
+ 45,
+ CONSTANTS.LOTTO_NUMBERS_COUNT
+ );
+ const lotto = new Lotto(numbers);
+ lottos.push(lotto);
+ }
+
+ return lottos;
+}
+
+export async function getWinningNumbersAndBonus() {
+ const winningNumbersInput = await Console.readLineAsync(
+ CONSTANTS.WINNING_NUMBER_INPUT_MESSAGE
+ );
+ const winningNums = winningNumbersInput
+ .split(",")
+ .map((number) => Number(number.trim()));
+ const bonusNum = await Console.readLineAsync(
+ CONSTANTS.BONUS_NUMBER_INPUT_MESSAGE
+ );
+ return { winningNums, bonusNum };
+}
+
+export function countResults(lottos, winningNums, bonusNum, statistics) {
+ const lottoCnt = lottos.length;
+ for (let i = 0; i < lottoCnt; i++) {
+ const res = lottos[i].result(winningNums, bonusNum);
+ countMatchingNum(res.cnt, statistics);
+ }
+ return statistics;
+}
+
+function countMatchingNum(cnt, statistics) {
+ switch (cnt) {
+ case 3:
+ statistics.fifth_prize++;
+ statistics.totalPrice += CONSTANTS.MATCH_3;
+ break;
+ case 4:
+ statistics.fourth_prize++;
+ statistics.totalPrice += CONSTANTS.MATCH_4;
+ break;
+ case 5:
+ if (res.hasBonus) {
+ statistics.second_prize++;
+ statistics.totalPrice += CONSTANTS.MATCH_5_WITH_BONUS;
+ break;
+ }
+ statistics.third_prize++;
+ statistics.totalPrice += CONSTANTS.MATCH_5;
+ break;
+ case 6:
+ statistics.first_prize++;
+ statistics.totalPrice += CONSTANTS.MATCH_6;
+ break;
+ default:
+ break;
+ }
+}
+
+export function printStatistics(purchaseAmount, statistics) {
+ const roundedProfitPercentage = (
+ (statistics.totalPrice / purchaseAmount) *
+ 100
+ ).toFixed(1);
+ Console.print(CONSTANTS.LOTTO_STATISTICS_HEADER);
+ Console.print(`${CONSTANTS.PRIZE_3_MATCH}${statistics.fifth_prize}개`);
+ Console.print(`${CONSTANTS.PRIZE_4_MATCH}${statistics.fourth_prize}개`);
+ Console.print(`${CONSTANTS.PRIZE_5_MATCH}${statistics.third_prize}개`);
+ Console.print(
+ `${CONSTANTS.PRIZE_5_MATCH_WITH_BONUS}${statistics.second_prize}개`
+ );
+ Console.print(`${CONSTANTS.PRIZE_6_MATCH}${statistics.first_prize}개`);
+ Console.print(
+ `${CONSTANTS.TOTAL_PROFIT_PERCENTAGE}${roundedProfitPercentage}%입니다.`
+ );
+}
From ebea5548374a52084f15de288c8e6cf7d56620c1 Mon Sep 17 00:00:00 2001
From: Taek Ki Min <127868094+TaekkiMin@users.noreply.github.com>
Date: Wed, 8 Nov 2023 23:45:25 +0900
Subject: [PATCH 6/8] =?UTF-8?q?app.js=20=ED=8C=8C=EC=9D=BC=20=EA=B5=AC?=
=?UTF-8?q?=ED=98=84?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
src/App.js | 36 +++++++++++++++++++++++++++++++++++-
1 file changed, 35 insertions(+), 1 deletion(-)
diff --git a/src/App.js b/src/App.js
index c38b30d5b2..c964e58bb9 100644
--- a/src/App.js
+++ b/src/App.js
@@ -1,5 +1,39 @@
+import {
+ countResults,
+ getValidPurchaseAmount,
+ getWinningNumbersAndBonus,
+ printStatistics,
+ purchaseLottos,
+} from "./LottoUtils.js";
+import * as CONSTANTS from "./Constants.js";
+
class App {
- async play() {}
+ constructor() {
+ this.purchaseAmount = 0;
+ this.lottos = [];
+ this.winningResult = { winningNums: [], bonusNum: 0 };
+ this.statistics = {
+ first_prize: 0,
+ second_prize: 0,
+ third_prize: 0,
+ fourth_prize: 0,
+ fifth_prize: 0,
+ totalPrice: 0,
+ };
+ }
+ async play() {
+ this.purchaseAmount = await getValidPurchaseAmount();
+ const lottoCnt = this.purchaseAmount / CONSTANTS.LOTTO_TICKET_PRICE;
+ this.lottos = purchaseLottos(lottoCnt);
+ this.winningResult = await getWinningNumbersAndBonus();
+ this.statistics = countResults(
+ this.lottos,
+ this.winningResult.winningNums,
+ this.winningResult.bonusNum,
+ this.statistics
+ );
+ printStatistics(this.purchaseAmount, this.statistics);
+ }
}
export default App;
From 319ab0188cae5584c4e4b6e030b5307f99af1011 Mon Sep 17 00:00:00 2001
From: Taek Ki Min <127868094+TaekkiMin@users.noreply.github.com>
Date: Wed, 8 Nov 2023 23:53:46 +0900
Subject: [PATCH 7/8] =?UTF-8?q?test=20=EC=84=B1=EA=B3=B5?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
__tests__/LottoTest.js | 18 ++++++++++++++++--
1 file changed, 16 insertions(+), 2 deletions(-)
diff --git a/__tests__/LottoTest.js b/__tests__/LottoTest.js
index 97bd457659..6642a835e2 100644
--- a/__tests__/LottoTest.js
+++ b/__tests__/LottoTest.js
@@ -1,5 +1,12 @@
+import { MissionUtils } from "@woowacourse/mission-utils";
import Lotto from "../src/Lotto.js";
+const getLogSpy = () => {
+ const logSpy = jest.spyOn(MissionUtils.Console, "print");
+ logSpy.mockClear();
+ return logSpy;
+};
+
describe("로또 클래스 테스트", () => {
test("로또 번호의 개수가 6개가 넘어가면 예외가 발생한다.", () => {
expect(() => {
@@ -7,12 +14,19 @@ describe("로또 클래스 테스트", () => {
}).toThrow("[ERROR]");
});
- // TODO: 이 테스트가 통과할 수 있게 구현 코드 작성
test("로또 번호에 중복된 숫자가 있으면 예외가 발생한다.", () => {
expect(() => {
new Lotto([1, 2, 3, 4, 5, 5]);
}).toThrow("[ERROR]");
});
- // 아래에 추가 테스트 작성 가능
+ test("로또 번호는 오름차순으로 정렬되어 출력해야 한다.", () => {
+ const logSpy = getLogSpy();
+
+ new Lotto([5, 3, 1, 4, 2, 6]);
+
+ const log = "[1, 2, 3, 4, 5, 6]";
+
+ expect(logSpy).toHaveBeenCalledWith(expect.stringContaining(log));
+ });
});
From caf171a6a14e8615ad2b0873c6e49d66d16efb01 Mon Sep 17 00:00:00 2001
From: Taek Ki Min <127868094+TaekkiMin@users.noreply.github.com>
Date: Wed, 8 Nov 2023 23:54:04 +0900
Subject: [PATCH 8/8] =?UTF-8?q?=EB=A6=AC=EB=93=9C=EB=AF=B8=20=EC=9E=91?=
=?UTF-8?q?=EC=84=B1?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
README.md | 259 +++++++++---------------------------------------------
1 file changed, 40 insertions(+), 219 deletions(-)
diff --git a/README.md b/README.md
index aff2c46597..e46f540817 100644
--- a/README.md
+++ b/README.md
@@ -1,239 +1,60 @@
# 미션 - 로또
-## 🔍 진행 방식
+## ✏️ 구현할 기능 목록
-- 미션은 **기능 요구 사항, 프로그래밍 요구 사항, 과제 진행 요구 사항** 세 가지로 구성되어 있다.
-- 세 개의 요구 사항을 만족하기 위해 노력한다. 특히 기능을 구현하기 전에 기능 목록을 만들고, 기능 단위로 커밋 하는 방식으로 진행한다.
-- 기능 요구 사항에 기재되지 않은 내용은 스스로 판단하여 구현한다.
+### `App.js`의 `play` 메서드를 통해 프로그램 시작
-## 📮 미션 제출 방법
+- [x] play 메서드 생성
-- 미션 구현을 완료한 후 GitHub을 통해 제출해야 한다.
- - GitHub을 활용한 제출 방법은 [프리코스 과제 제출](https://github.com/woowacourse/woowacourse-docs/tree/master/precourse) 문서를 참고해
- 제출한다.
-- GitHub에 미션을 제출한 후 [우아한테크코스 지원](https://apply.techcourse.co.kr) 사이트에 접속하여 프리코스 과제를 제출한다.
- - 자세한 방법은 [제출 가이드](https://github.com/woowacourse/woowacourse-docs/tree/master/precourse#제출-가이드) 참고
- - **Pull Request만 보내고 지원 플랫폼에서 과제를 제출하지 않으면 최종 제출하지 않은 것으로 처리되니 주의한다.**
+### 로또 로직
-## 🚨 과제 제출 전 체크 리스트 - 0점 방지
+- [x] 로또 구입 금액 입력
-- 기능 구현을 모두 정상적으로 했더라도 **요구 사항에 명시된 출력값 형식을 지키지 않을 경우 0점으로 처리**한다.
-- 기능 구현을 완료한 뒤 아래 가이드에 따라 테스트를 실행했을 때 모든 테스트가 성공하는지 확인한다.
-- **테스트가 실패할 경우 0점으로 처리**되므로, 반드시 확인 후 제출한다.
+ - 구입 금액 입력 안내 문구 출력
+ `구입금액을 입력해 주세요.`
-### 테스트 실행 가이드
+ - 잘못된 값 입력 시 **throw문을 통해 예외 발생 및 애플리케이션 종료**
+ - 1,000원으로 나누어 떨어지지 않는 경우
-- 테스트 패키지 설치를 위해 `Node.js` 버전 `18.17.1` 이상이 필요하다.
-- 다음 명령어를 입력해 패키지를 설치한다.
+- [x] 구입 금액에 해당하는 만큼 로또 발행(로또 1장 당 1,000원)
-```bash
-npm install
-```
+ - 구매 갯수 문구 출력
+ `${로또 구입 금액/1000}개를 구매했습니다.`
-- 설치가 완료되었다면, 다음 명령어를 입력해 테스트를 실행한다.
+ - 각 로또 번호 출력(구매 갯수 만큼)
+ - `Random API`를 이용하여 1-45 사이 값 6개 선택
+ - 로또 번호는 오름차순으로 정렬하여 보여준다.
-```bash
-npm test
-```
+- [x] 당첨 번호 입력
----
+ - 당첨 번호 입력 안내 문구 출력
+ `당첨 번호를 입력해 주세요.`
-## 🚀 기능 요구 사항
+ - 잘못된 값 입력 시 **throw문을 통해 예외 발생 및 애플리케이션 종료**
+ - 쉼표로 구분했을 때, 번호 6개
+ - 각 번호가 1-45 범위 내의 수가 아닌 경우
-- 로또 게임 기능을 구현해야 한다. 로또 게임은 아래와 같은 규칙으로 진행된다.
+- [x] 보너스 번호 입력
-```
-- 로또 번호의 숫자 범위는 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원이다.
-- 당첨 번호와 보너스 번호를 입력받는다.
-- 사용자가 구매한 로또 번호와 당첨 번호를 비교하여 당첨 내역 및 수익률을 출력하고 로또 게임을 종료한다.
-- 사용자가 잘못된 값을 입력할 경우 `throw`문을 사용해 예외를 발생시킨다. 그런 다음, "[ERROR]"로 시작하는 에러 메시지를 출력하고 해당 부분부터 입력을 다시 받는다.
- ```
- 예시) [ERROR] 숫자가 잘못된 형식입니다.
- ```
-
-### 입출력 요구 사항
-
-#### 입력
-
-- 로또 구입 금액을 입력 받는다. 구입 금액은 1,000원 단위로 입력 받으며 1,000원으로 나누어 떨어지지 않는 경우 예외 처리한다.
-
-```
-14000
-```
-
-- 당첨 번호를 입력 받는다. 번호는 쉼표(,)를 기준으로 구분한다.
-
-```
-1,2,3,4,5,6
-```
-
-- 보너스 번호를 입력 받는다.
-
-```
-7
-```
-
-#### 출력
-
-- 발행한 로또 수량 및 번호를 출력한다. 로또 번호는 오름차순으로 정렬하여 보여준다.
-
-```
-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]
-```
-
-- 당첨 내역을 출력한다.
-
-```
-3개 일치 (5,000원) - 1개
-4개 일치 (50,000원) - 0개
-5개 일치 (1,500,000원) - 0개
-5개 일치, 보너스 볼 일치 (30,000,000원) - 0개
-6개 일치 (2,000,000,000원) - 0개
-```
+ - 잘못된 값 입력 시 **throw문을 통해 예외 발생 및 애플리케이션 종료**
+ - 번호가 1-45 범위 내의 수가 아닌 경우
-- 수익률은 소수점 둘째 자리에서 반올림한다. (ex. 100.0%, 51.5%, 1,000,000.0%)
+- [x] 로또 결과 출력
-```
-총 수익률은 62.5%입니다.
-```
+ - 당첨 내역 출력
+ - 수익률은 소수점 둘째 자리에서 반올림
+ - 아래는 예시이다.
-- 예외 상황 시 에러 문구를 출력해야 한다. 단, 에러 문구는 "[ERROR]"로 시작해야 한다.
-
-```
-[ERROR] 로또 번호는 1부터 45 사이의 숫자여야 합니다.
-```
-
-#### 실행 결과 예시
-
-```
-구입금액을 입력해 주세요.
-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%입니다.
-```
-
----
-
-## 🎯 프로그래밍 요구 사항
-
-- Node.js 18.17.1 버전에서 실행 가능해야 한다. **Node.js 18.17.1에서 정상적으로 동작하지 않을 경우 0점 처리한다.**
-- 프로그램 실행의 시작점은 `App.js`의 `play` 메서드이다. 아래와 같이 프로그램을 실행시킬 수 있어야 한다.
-
-**예시**
-
-```javascript
-const app = new App();
-app.play();
-```
-
-- `package.json`을 변경할 수 없고 외부 라이브러리(jQuery, Lodash 등)를 사용하지 않는다. 순수 Vanilla JS로만 구현한다.
-- [JavaScript 코드 컨벤션](https://github.com/woowacourse/woowacourse-docs/tree/main/styleguide/javascript)을 지키면서 프로그래밍 한다
-- 프로그램 종료 시 `process.exit()`를 호출하지 않는다.
-- 프로그램 구현이 완료되면 `ApplicationTest`의 모든 테스트가 성공해야 한다. **테스트가 실패할 경우 0점 처리한다.**
-- 프로그래밍 요구 사항에서 달리 명시하지 않는 한 파일, 패키지 이름을 수정하거나 이동하지 않는다.
-- indent(인덴트, 들여쓰기) depth를 3이 넘지 않도록 구현한다. 2까지만 허용한다.
- - 예를 들어 while문 안에 if문이 있으면 들여쓰기는 2이다.
- - 힌트: indent(인덴트, 들여쓰기) depth를 줄이는 좋은 방법은 함수(또는 메서드)를 분리하면 된다.
-- 함수(또는 메서드)가 한 가지 일만 하도록 최대한 작게 만들어라.
-- Jest를 이용하여 본인이 정리한 기능 목록이 정상 동작함을 테스트 코드로 확인한다.
-
-### 추가된 요구 사항
-
-- 함수(또는 메서드)의 길이가 15라인을 넘어가지 않도록 구현한다.
- - 함수(또는 메서드)가 한 가지 일만 잘 하도록 구현한다.
-- else를 지양한다.
- - 힌트: if 조건절에서 값을 return하는 방식으로 구현하면 else를 사용하지 않아도 된다.
- - 때로는 if/else, switch문을 사용하는 것이 더 깔끔해 보일 수 있다. 어느 경우에 쓰는 것이 적절할지 스스로 고민해 본다.
-- 도메인 로직에 단위 테스트를 구현해야 한다. 단, UI(Console.readLineAsync, Console.print) 로직에 대한 단위 테스트는 제외한다.
- - 핵심 로직을 구현하는 코드와 UI를 담당하는 로직을 구분한다.
- - 단위 테스트 작성이 익숙하지 않다면 `__tests__/LottoTest.js`를 참고하여 학습한 후 테스트를 구현한다.
-
-### 라이브러리
-
-- `@woowacourse/mission-utils`에서 제공하는 `Random` 및 `Console` API를 사용하여 구현해야 한다.
- - Random 값 추출은 `Random.pickUniqueNumbersInRange()`를 활용한다.
- - 사용자의 값을 입력 받고 출력하기 위해서는 `Console.readLineAsync`, `Console.print`를 활용한다.
-
-#### 사용 예시
-
-```javascript
-MissionUtils.Random.pickUniqueNumbersInRange(1, 45, 6);
-```
-
-### Lotto 클래스
-
-- 제공된 `Lotto` 클래스를 활용해 구현해야 한다.
-- `numbers`의 `#` prefix를 변경할 수 없다.
-- `Lotto`에 필드를 추가할 수 없다.
-
-```javascript
-class Lotto {
- #numbers;
-
- constructor(numbers) {
- this.#validate(numbers);
- this.#numbers = numbers;
- }
-
- #validate(numbers) {
- if (numbers.length !== 6) {
- throw new Error("[ERROR] 로또 번호는 6개여야 합니다.");
- }
- }
-
- // TODO: 추가 기능 구현
-}
-```
-
----
-
-## ✏️ 과제 진행 요구 사항
-
-- 미션은 [javascript-lotto-6](https://github.com/woowacourse-precourse/javascript-lotto-6/) 저장소를 Fork & Clone해 시작한다.
-- **기능을 구현하기 전 `docs/README.md`에 구현할 기능 목록을 정리**해 추가한다.
-- **Git의 커밋 단위는 앞 단계에서 `docs/README.md`에 정리한 기능 목록 단위**로 추가한다.
- - [커밋 메시지 컨벤션](https://gist.github.com/stephenparish/9941e89d80e2bc58a153) 가이드를 참고해 커밋 메시지를 작성한다.
-- 과제 진행 및 제출 방법은 [프리코스 과제 제출](https://github.com/woowacourse/woowacourse-docs/tree/master/precourse) 문서를 참고한다.
+ ```
+ 당첨 통계
+ ---
+ 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%입니다.
+ ```