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

[로또] 민택기 미션 제출합니다. #637

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
259 changes: 40 additions & 219 deletions README.md
Original file line number Diff line number Diff line change
@@ -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점으로 처리**되므로, 반드시 확인 후 제출한다.
- 구입 금액 입력 안내 문구 출력<br>
`구입금액을 입력해 주세요.`

### 테스트 실행 가이드
- 잘못된 값 입력 시 **throw문을 통해 예외 발생 및 애플리케이션 종료**<br>
- 1,000원으로 나누어 떨어지지 않는 경우

- 테스트 패키지 설치를 위해 `Node.js` 버전 `18.17.1` 이상이 필요하다.
- 다음 명령어를 입력해 패키지를 설치한다.
- [x] 구입 금액에 해당하는 만큼 로또 발행(로또 1장 당 1,000원)

```bash
npm install
```
- 구매 갯수 문구 출력<br>
`${로또 구입 금액/1000}개를 구매했습니다.`

- 설치가 완료되었다면, 다음 명령어를 입력해 테스트를 실행한다.
- 각 로또 번호 출력(구매 갯수 만큼)
- `Random API`를 이용하여 1-45 사이 값 6개 선택
- 로또 번호는 오름차순으로 정렬하여 보여준다.

```bash
npm test
```
- [x] 당첨 번호 입력

---
- 당첨 번호 입력 안내 문구 출력<br>
`당첨 번호를 입력해 주세요.`

## 🚀 기능 요구 사항
- 잘못된 값 입력 시 **throw문을 통해 예외 발생 및 애플리케이션 종료**<br>
- 쉼표로 구분했을 때, 번호 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원
```
- 보너스 번호 입력 안내 문구 출력<br>
`보너스 번호를 입력해 주세요.`

- 로또 구입 금액을 입력하면 구입 금액에 해당하는 만큼 로또를 발행해야 한다.
- 로또 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문을 통해 예외 발생 및 애플리케이션 종료**<br>
- 번호가 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%입니다.
```
18 changes: 16 additions & 2 deletions __tests__/LottoTest.js
Original file line number Diff line number Diff line change
@@ -1,18 +1,32 @@
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(() => {
new Lotto([1, 2, 3, 4, 5, 6, 7]);
}).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));
});
});
36 changes: 35 additions & 1 deletion src/App.js
Original file line number Diff line number Diff line change
@@ -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;
30 changes: 30 additions & 0 deletions src/Constants.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
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;
Loading