From a5dfcfd2ebdeafd487d2a09d6af0968f61683fa3 Mon Sep 17 00:00:00 2001 From: syg0629 Date: Mon, 4 Nov 2024 20:37:05 +0900 Subject: [PATCH 01/11] =?UTF-8?q?docs(readme):=20=EA=B5=AC=ED=98=84?= =?UTF-8?q?=ED=95=A0=20=EA=B8=B0=EB=8A=A5=20=EB=AA=A9=EB=A1=9D=20=EB=B0=8F?= =?UTF-8?q?=20=EC=98=88=EC=99=B8=20=EC=BC=80=EC=9D=B4=EC=8A=A4=20=EC=A0=95?= =?UTF-8?q?=EB=A6=AC?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- README.md | 222 ++++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 222 insertions(+) diff --git a/README.md b/README.md index 15bb106b5..09996d3c1 100644 --- a/README.md +++ b/README.md @@ -1 +1,223 @@ # javascript-lotto-precourse + +## 구현할 기능 목록 + +1. 사용자 입력 처리 + - 구입 금액 + - 당첨 번호 + - 보너스 번호 +2. 로또 생성 + - 구입 금액에 따라 구매 가능한 로또의 개수 계산 + - 로또 개수에 따라 로또 번호 자동 생성, 오름차순으로 정렬 +3. 당첨 확인 + - 구매한 로또와 당첨 번호 비교 + - 일치하는 번호 개수 확인 + - 보너스 번호 일치 여부 확인 + - 당첨 등수 판정 +4. 결과 출력 + - 당첨 내역 출력(각 등수별 당첨 개수) + - 수익률 계산 + +## 예외 케이스 + +### 구입금액 + +1. 숫자가 아닐 때 +2. 실수일 때 +3. 공백일 때 +4. 1000원으로 나누어 떨어지지 않을 때 + +### 로또 번호 + +1. 숫자가 아닐 때 +2. 실수일 때 +3. 공백일 때 +4. 1 ~ 45번까지의 숫자가 아닐 때 + +--- + +--- + +## 기능 요구 사항 + +간단한 로또 발매기를 구현한다. + +- 로또 번호의 숫자 범위는 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를 발생시키고 해당 메시지를 출력한 다음 해당 지점부터 다시 입력을 받는다. + +### 입출력 요구 사항 + +#### 입력 + +- 로또 구입 금액을 입력 받는다. 구입 금액은 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개 +``` + +- 수익률은 소수점 둘째 자리에서 반올림한다. (ex. 100.0%, 51.5%, 1,000,000.0%) + +``` +총 수익률은 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%입니다. +``` + +## 프로그래밍 요구 사항 1 + +- Node.js 20.17.0 버전에서 실행 가능해야 한다. +- 프로그램 실행의 시작점은 App.js의 run()이다. +- package.json 파일은 변경할 수 없으며, 제공된 라이브러리와 스타일 라이브러리 이외의 외부 라이브러리는 사용하지 않는다. +- 프로그램 종료 시 process.exit()를 호출하지 않는다. +- 프로그래밍 요구 사항에서 달리 명시하지 않는 한 파일, 패키지 등의 이름을 바꾸거나 이동하지 않는다. +- 자바스크립트 코드 컨벤션을 지키면서 프로그래밍한다. + - 기본적으로 JavaScript Style Guide를 원칙으로 한다. + +## 프로그래밍 요구 사항 2 + +- indent(인덴트, 들여쓰기) depth를 3이 넘지 않도록 구현한다. 2까지만 허용한다. + - 예를 들어 while문 안에 if문이 있으면 들여쓰기는 2이다. + - 힌트: indent(인덴트, 들여쓰기) depth를 줄이는 좋은 방법은 함수(또는 메서드)를 분리하면 된다. +- 3항 연산자를 쓰지 않는다. +- 함수(또는 메서드)가 한 가지 일만 하도록 최대한 작게 만들어라. +- Jest를 이용하여 정리한 기능 목록이 정상적으로 작동하는지 테스트 코드로 확인한다. + - 테스트 도구 사용법이 익숙하지 않다면 아래 문서를 참고하여 학습한 후 테스트를 구현한다. + - Using Matchers + - Testing Asynchronous Code + - Jest로 파라미터화 테스트하기: test.each(), describe.each() + +## 프로그래밍 요구 사항 3 + +- 함수(또는 메서드)의 길이가 15라인을 넘어가지 않도록 구현한다. + - 함수(또는 메서드)가 한 가지 일만 잘 하도록 구현한다. +- else를 지양한다. + - 때로는 if/else, when문을 사용하는 것이 더 깔끔해 보일 수 있다. 어느 경우에 쓰는 것이 적절할지 스스로 고민해 본다. + - 힌트: if 조건절에서 값을 return하는 방식으로 구현하면 else를 사용하지 않아도 된다. +- 구현한 기능에 대한 단위 테스트를 작성한다. 단, UI(System.out, System.in, Scanner) 로직은 제외한다. + - 단위 테스트 작성이 익숙하지 않다면 LottoTest를 참고하여 학습한 후 테스트를 작성한다. + +### 라이브러리 + +- @woowacourse/mission-utils에서 제공하는 Random 및 Console API를 사용하여 구현해야 한다. + - Random 값 추출은 Random. pickUniqueNumbersInRange()를 활용한다. + - 사용자의 값을 입력 및 출력하려면 Console.readLineAsync()와 Console.print()를 활용한다. + +#### 사용 예시 + +- 1에서 45 사이의 중복되지 않은 정수 6개 반환 + +``` +MissionUtils.Random.pickUniqueNumbersInRange(1, 45, 6); +``` + +### Lotto 클래스 + +- 제공된 Lotto 클래스를 사용하여 구현해야 한다. +- Lotto에 numbers 이외의 필드(인스턴스 변수)를 추가할 수 없다. +- numbers의 접근 제어자인 #은 변경할 수 없다. +- Lotto의 패키지를 변경할 수 있다. + +``` +class Lotto { + #numbers; + + constructor(numbers) { + this.#validate(numbers); + this.#numbers = numbers; + } + + #validate(numbers) { + if (numbers.length !== 6) { + throw new Error("[ERROR] 로또 번호는 6개여야 합니다."); + } + } + + // TODO: 추가 기능 구현 +} +``` From 90db10a6b07fd763845a0a7cd6f82d0cffd6bcce Mon Sep 17 00:00:00 2001 From: syg0629 Date: Mon, 4 Nov 2024 20:44:22 +0900 Subject: [PATCH 02/11] =?UTF-8?q?feat(useInput):=20=EC=82=AC=EC=9A=A9?= =?UTF-8?q?=EC=9E=90=20=EC=9E=85=EB=A0=A5=20=EC=B2=98=EB=A6=AC=20-=20?= =?UTF-8?q?=EA=B5=AC=EC=9E=85=EA=B8=88=EC=95=A1=20-=20=EB=8B=B9=EC=B2=A8?= =?UTF-8?q?=20=EB=B2=88=ED=98=B8=20-=20=EB=B3=B4=EB=84=88=EC=8A=A4=20?= =?UTF-8?q?=EB=B2=88=ED=98=B8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/App.js | 14 +++++++++++++- src/UserInput.js | 20 ++++++++++++++++++++ 2 files changed, 33 insertions(+), 1 deletion(-) create mode 100644 src/UserInput.js diff --git a/src/App.js b/src/App.js index 091aa0a5d..d4e933c73 100644 --- a/src/App.js +++ b/src/App.js @@ -1,5 +1,17 @@ +import UserInput from "./UserInput.js"; + class App { - async run() {} + #userInput; + + constructor() { + this.#userInput = new UserInput(); + } + + async run() { + const amount = await this.#userInput.getUserInput("purchaseAmount"); + const numbers = await this.#userInput.getUserInput("winningNumber"); + const bonus = await this.#userInput.getUserInput("bonusNumber"); + } } export default App; diff --git a/src/UserInput.js b/src/UserInput.js new file mode 100644 index 000000000..698f0690d --- /dev/null +++ b/src/UserInput.js @@ -0,0 +1,20 @@ +import { MissionUtils } from "@woowacourse/mission-utils"; + +const INPUT_MESSAGE = { + purchaseAmount: "구입금액을 입력해 주세요.\n", + winningNumber: "당첨 번호를 입력해 주세요.\n", + bonusNummber: "보너스 번호를 입력해 주세요.\n", +}; + +class UserInput { + async getUserInput(messageType) { + if (!INPUT_MESSAGE[messageType]) + throw new Error("[Error] 존재하지 않는 입력 메세지 입니다."); + const input = await MissionUtils.Console.readLineAsync( + INPUT_MESSAGE[messageType] + ); + return input; + } +} + +export default UserInput; From 38f114fd0a3476c323582d6ec847abdefdd05b1f Mon Sep 17 00:00:00 2001 From: syg0629 Date: Mon, 4 Nov 2024 22:44:14 +0900 Subject: [PATCH 03/11] =?UTF-8?q?style:=20=EC=9E=85=EB=A0=A5=20=EB=A9=94?= =?UTF-8?q?=EC=8B=9C=EC=A7=80=EC=97=90=20=EC=A4=84=EB=B0=94=EA=BF=88=20?= =?UTF-8?q?=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/UserInput.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/UserInput.js b/src/UserInput.js index 698f0690d..13613aa66 100644 --- a/src/UserInput.js +++ b/src/UserInput.js @@ -2,8 +2,8 @@ import { MissionUtils } from "@woowacourse/mission-utils"; const INPUT_MESSAGE = { purchaseAmount: "구입금액을 입력해 주세요.\n", - winningNumber: "당첨 번호를 입력해 주세요.\n", - bonusNummber: "보너스 번호를 입력해 주세요.\n", + winningNumber: "\n당첨 번호를 입력해 주세요.\n", + bonusNummber: "\n보너스 번호를 입력해 주세요.\n", }; class UserInput { From 666700b9ce09ad5a6c889419755df77fa297be57 Mon Sep 17 00:00:00 2001 From: syg0629 Date: Mon, 4 Nov 2024 22:46:52 +0900 Subject: [PATCH 04/11] =?UTF-8?q?feat(lottoCreator):=20=EB=A1=9C=EB=98=90?= =?UTF-8?q?=20=EC=83=9D=EC=84=B1=20-=20=EA=B5=AC=EC=9E=85=20=EA=B8=88?= =?UTF-8?q?=EC=95=A1=EC=97=90=20=EB=94=B0=EB=A5=B8=20=EA=B5=AC=EB=A7=A4=20?= =?UTF-8?q?=EA=B0=80=EB=8A=A5=ED=95=9C=20=EB=A1=9C=EB=98=90=20=EA=B0=9C?= =?UTF-8?q?=EC=88=98=20=EA=B3=84=EC=82=B0=20-=20=EB=A1=9C=EB=98=90=20?= =?UTF-8?q?=EA=B0=9C=EC=88=98=EC=97=90=20=EB=94=B0=EB=9D=BC=20=EB=A1=9C?= =?UTF-8?q?=EB=98=90=20=EB=B2=88=ED=98=B8=20=EC=9E=90=EB=8F=99=20=EC=83=9D?= =?UTF-8?q?=EC=84=B1=20-=20=EC=83=9D=EC=84=B1=EB=90=9C=20=EB=A1=9C?= =?UTF-8?q?=EB=98=90=20=EC=B6=9C=EB=A0=A5?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/LottoCreator.js | 31 +++++++++++++++++++++++++++++++ 1 file changed, 31 insertions(+) create mode 100644 src/LottoCreator.js diff --git a/src/LottoCreator.js b/src/LottoCreator.js new file mode 100644 index 000000000..9dd692978 --- /dev/null +++ b/src/LottoCreator.js @@ -0,0 +1,31 @@ +import { MissionUtils } from "@woowacourse/mission-utils"; +import Lotto from "./Lotto.js"; + +const LOTTO_PRICE = 1000; + +class LottoCreator { + getLottoCount(amount) { + return Math.floor(amount / LOTTO_PRICE); + } + + createLotto(lottoCount) { + const lottos = []; + for (let i = 0; i < lottoCount; i++) { + const lottoNumber = MissionUtils.Random.pickUniqueNumbersInRange( + 1, + 45, + 6 + ); + lottos.push(new Lotto(lottoNumber)); + } + return lottos; + } + + printLottos(lottos) { + lottos.forEach((lotto) => { + MissionUtils.Console.print(lotto.getNumbers()); + }); + } +} + +export default LottoCreator; From 3ea5a4373f87235055bbfb84f4ca5414ed8f4bdb Mon Sep 17 00:00:00 2001 From: syg0629 Date: Mon, 4 Nov 2024 22:47:41 +0900 Subject: [PATCH 05/11] =?UTF-8?q?feat(lotto):=20=EB=A1=9C=EB=98=90=20?= =?UTF-8?q?=EC=98=A4=EB=A6=84=EC=B0=A8=EC=88=9C=20=EC=A0=95=EB=A0=AC,=20?= =?UTF-8?q?=EC=9C=A0=ED=9A=A8=EC=84=B1=20=EA=B2=80=EC=82=AC=20=EC=B6=94?= =?UTF-8?q?=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/Lotto.js | 14 ++++++++++++-- 1 file changed, 12 insertions(+), 2 deletions(-) diff --git a/src/Lotto.js b/src/Lotto.js index cb0b1527e..995f86c63 100644 --- a/src/Lotto.js +++ b/src/Lotto.js @@ -3,16 +3,26 @@ class Lotto { constructor(numbers) { this.#validate(numbers); - this.#numbers = numbers; + this.#numbers = numbers.sort((a, b) => a - b); } #validate(numbers) { if (numbers.length !== 6) { throw new Error("[ERROR] 로또 번호는 6개여야 합니다."); } + + if (!numbers.every((number) => number >= 1 && number <= 45)) { + throw new Error("[ERROR] 로또 번호는 1 ~ 45 사이의 숫자여야 합니다."); + } + + if (new Set(numbers).size !== 6) { + throw new Error("[ERROR] 로또 번호는 중복되지 않아야 합니다."); + } } - // TODO: 추가 기능 구현 + getNumbers() { + return [...this.#numbers]; + } } export default Lotto; From e4f22e1549dee80f6c14ea7c4f119d3d23a6baee Mon Sep 17 00:00:00 2001 From: syg0629 Date: Mon, 4 Nov 2024 22:48:20 +0900 Subject: [PATCH 06/11] =?UTF-8?q?docs(readme):=20=EC=98=88=EC=99=B8=20?= =?UTF-8?q?=EC=BC=80=EC=9D=B4=EC=8A=A4=20=EC=B6=94=EA=B0=80=20-=20?= =?UTF-8?q?=EB=A1=9C=EB=98=90=20=EB=B2=88=ED=98=B8=EA=B0=80=20=EC=A4=91?= =?UTF-8?q?=EB=B3=B5=EC=9D=BC=20=EA=B2=BD=EC=9A=B0?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- README.md | 1 + 1 file changed, 1 insertion(+) diff --git a/README.md b/README.md index 09996d3c1..84da14c5d 100644 --- a/README.md +++ b/README.md @@ -33,6 +33,7 @@ 2. 실수일 때 3. 공백일 때 4. 1 ~ 45번까지의 숫자가 아닐 때 +5. 중복 번호 없어야함 --- From 228ac5a94d667d1d8433073eb3f1b7bf6cd16303 Mon Sep 17 00:00:00 2001 From: syg0629 Date: Mon, 4 Nov 2024 22:49:06 +0900 Subject: [PATCH 07/11] =?UTF-8?q?feat(app):=20=EB=A1=9C=EB=98=90=20?= =?UTF-8?q?=EC=83=9D=EC=84=B1=20=EB=B0=8F=20=EC=B6=9C=EB=A0=A5=20=EB=B6=80?= =?UTF-8?q?=EB=B6=84=20=EC=B6=94=EA=B0=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/App.js | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/src/App.js b/src/App.js index d4e933c73..39b9938cc 100644 --- a/src/App.js +++ b/src/App.js @@ -1,14 +1,25 @@ +import LottoCreator from "./LottoCreator.js"; import UserInput from "./UserInput.js"; +import { MissionUtils } from "@woowacourse/mission-utils"; class App { #userInput; + #lottoCreator; constructor() { this.#userInput = new UserInput(); + this.#lottoCreator = new LottoCreator(); } async run() { const amount = await this.#userInput.getUserInput("purchaseAmount"); + const lottoCount = this.#lottoCreator.getLottoCount(amount); + + MissionUtils.Console.print(`\n${lottoCount}개를 구매했습니다.`); + + const lottos = this.#lottoCreator.createLotto(lottoCount); + this.#lottoCreator.printLottos(lottos); + const numbers = await this.#userInput.getUserInput("winningNumber"); const bonus = await this.#userInput.getUserInput("bonusNumber"); } From ae1240a4dd9d0af700abe82fa01a0f6b3f3917ce Mon Sep 17 00:00:00 2001 From: syg0629 Date: Mon, 4 Nov 2024 23:57:12 +0900 Subject: [PATCH 08/11] =?UTF-8?q?style:=20=EC=98=A4=ED=83=80=20=EC=88=98?= =?UTF-8?q?=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/UserInput.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/UserInput.js b/src/UserInput.js index 13613aa66..500ca405e 100644 --- a/src/UserInput.js +++ b/src/UserInput.js @@ -3,7 +3,7 @@ import { MissionUtils } from "@woowacourse/mission-utils"; const INPUT_MESSAGE = { purchaseAmount: "구입금액을 입력해 주세요.\n", winningNumber: "\n당첨 번호를 입력해 주세요.\n", - bonusNummber: "\n보너스 번호를 입력해 주세요.\n", + bonusNumber: "\n보너스 번호를 입력해 주세요.\n", }; class UserInput { From b8222f8995aacf74305554db3e766146e5d6b83b Mon Sep 17 00:00:00 2001 From: syg0629 Date: Mon, 4 Nov 2024 23:59:40 +0900 Subject: [PATCH 09/11] =?UTF-8?q?feat(lottoChecker):=20=EB=8B=B9=EC=B2=A8?= =?UTF-8?q?=20=ED=99=95=EC=9D=B8=20-=20=EA=B5=AC=EB=A7=A4=ED=95=9C=20?= =?UTF-8?q?=EB=A1=9C=EB=98=90=EC=99=80=20=EB=8B=B9=EC=B2=A8=20=EB=B2=88?= =?UTF-8?q?=ED=98=B8=20=EB=B9=84=EA=B5=90=20-=20=EC=9D=BC=EC=B9=98?= =?UTF-8?q?=ED=95=98=EB=8A=94=20=EB=B2=88=ED=98=B8=20=EA=B0=9C=EC=88=98=20?= =?UTF-8?q?=ED=99=95=EC=9D=B8=20-=20=EB=B3=B4=EB=84=88=EC=8A=A4=20?= =?UTF-8?q?=EB=B2=88=ED=98=B8=20=EC=9D=BC=EC=B9=98=20=EC=97=AC=EB=B6=80=20?= =?UTF-8?q?=ED=99=95=EC=9D=B8=20-=20=EB=8B=B9=EC=B2=A8=20=EB=93=B1?= =?UTF-8?q?=EC=88=98=20=ED=8C=90=EC=A0=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/LottoChecker.js | 46 +++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 46 insertions(+) create mode 100644 src/LottoChecker.js diff --git a/src/LottoChecker.js b/src/LottoChecker.js new file mode 100644 index 000000000..025608010 --- /dev/null +++ b/src/LottoChecker.js @@ -0,0 +1,46 @@ +class LottoChecker { + #winningResult; + + constructor() { + this.#winningResult = { + 3: { prize: 5000, count: 0 }, + 4: { prize: 50000, count: 0 }, + 5: { prize: 1500000, count: 0 }, + 5.5: { prize: 30000000, count: 0 }, + 6: { prize: 2000000000, count: 0 }, + }; + } + + checkWinning(lottos, winningNumber, bonusNumber) { + lottos.forEach((lotto) => { + const matchCount = this.#countMatches(lotto.getNumbers(), winningNumber); + this.#countPrize(matchCount, lotto.getNumbers(), bonusNumber); + }); + return this.#winningResult; + } + + #countMatches(lottoNumbers, winningNumbers) { + return lottoNumbers.filter((number) => winningNumbers.includes(number)) + .length; + } + + #countPrize(matchCount, lottoNumbers, bonusNumber) { + if (matchCount === 5 && lottoNumbers.includes(bonusNumber)) { + this.#winningResult[5.5].count += 1; + return; + } + + if (this.#winningResult[matchCount]) { + this.#winningResult[matchCount].count += 1; + } + } + + calculateTotalPrize() { + return Object.values(this.#winningResult).reduce( + (total, { prize, count }) => total + prize * count, + 0 + ); + } +} + +export default LottoChecker; From f503099ca97054f1377168f12d522b2b260bcb10 Mon Sep 17 00:00:00 2001 From: syg0629 Date: Mon, 4 Nov 2024 23:59:56 +0900 Subject: [PATCH 10/11] =?UTF-8?q?feat(outputview):=20=EC=88=98=EC=9D=B5?= =?UTF-8?q?=EB=A5=A0=20=ED=99=95=EC=9D=B8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/OutputView.js | 29 +++++++++++++++++++++++++++++ 1 file changed, 29 insertions(+) create mode 100644 src/OutputView.js diff --git a/src/OutputView.js b/src/OutputView.js new file mode 100644 index 000000000..2139b55a4 --- /dev/null +++ b/src/OutputView.js @@ -0,0 +1,29 @@ +import { MissionUtils } from "@woowacourse/mission-utils"; + +class OutputView { + printWinningStatistics(winningResult) { + MissionUtils.Console.print("\n당첨 통계\n---"); + MissionUtils.Console.print( + `3개 일치 (5,000원) - ${winningResult[3].count}개` + ); + MissionUtils.Console.print( + `4개 일치 (50,000원) - ${winningResult[4].count}개` + ); + MissionUtils.Console.print( + `5개 일치 (1,500,000원) - ${winningResult[5].count}개` + ); + MissionUtils.Console.print( + `5개 일치, 보너스 볼 일치 (30,000,000원) - ${winningResult[5.5].count}개` + ); + MissionUtils.Console.print( + `6개 일치 (2,000,000,000원) - ${winningResult[6].count}개` + ); + } + + printProfitRate(totalPrize, lottoCount) { + const profitRate = ((totalPrize / (lottoCount * 1000)) * 100).toFixed(1); + MissionUtils.Console.print(`총 수익률은 ${profitRate}%입니다.`); + } +} + +export default OutputView; From 68e25f7c67257f99a6002de4a2d92cf27ddc8990 Mon Sep 17 00:00:00 2001 From: syg0629 Date: Tue, 5 Nov 2024 00:00:16 +0900 Subject: [PATCH 11/11] =?UTF-8?q?feat(app):=20=EA=B2=B0=EA=B3=BC=20?= =?UTF-8?q?=EC=B6=9C=EB=A0=A5?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/App.js | 25 +++++++++++++++++++++++-- 1 file changed, 23 insertions(+), 2 deletions(-) diff --git a/src/App.js b/src/App.js index 39b9938cc..a7401a533 100644 --- a/src/App.js +++ b/src/App.js @@ -1,14 +1,20 @@ +import LottoChecker from "./LottoChecker.js"; import LottoCreator from "./LottoCreator.js"; +import OutputView from "./OutputView.js"; import UserInput from "./UserInput.js"; import { MissionUtils } from "@woowacourse/mission-utils"; class App { #userInput; #lottoCreator; + #lottoChecker; + #outputView; constructor() { this.#userInput = new UserInput(); this.#lottoCreator = new LottoCreator(); + this.#lottoChecker = new LottoChecker(); + this.#outputView = new OutputView(); } async run() { @@ -20,8 +26,23 @@ class App { const lottos = this.#lottoCreator.createLotto(lottoCount); this.#lottoCreator.printLottos(lottos); - const numbers = await this.#userInput.getUserInput("winningNumber"); - const bonus = await this.#userInput.getUserInput("bonusNumber"); + const winningNumbers = (await this.#userInput.getUserInput("winningNumber")) + .split(",") + .map((number) => Number(number.trim())); + const bonusNumber = Number( + await this.#userInput.getUserInput("bonusNumber") + ); + + const checkedLottos = this.#lottoChecker.checkWinning( + lottos, + winningNumbers, + bonusNumber + ); + + this.#outputView.printWinningStatistics(checkedLottos); + + const totalPrize = this.#lottoChecker.calculateTotalPrize(); + this.#outputView.printProfitRate(totalPrize, lottoCount); } }