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

[자동차 경주] 김다슬 미션 제출합니다. #445

Open
wants to merge 6 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
81 changes: 81 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
@@ -1 +1,82 @@
# javascript-racingcar-precourse
## 우아한 테크코스 2주차
### 자동차 경주

------

#### 과제 진행 요구사항

미션은 자동자 경주 저장소를 포크하고 클론하는 것으로 시작한다.

기능을 구현하기 전 `README.md`에 구현할 기능 목록을 정리해 추가한다.

------

#### 기능 요구 사항

초간단 자동차 경주 게임을 구현한다.

* 주어진 횟수 동안 n대의 자동차는 전진 또는 멈출 수 있다

* 각 자동차에 이름을 부여할 수 있다. 전진하는 자동차를 출력할 때 자동차 이름을 같이 출력한다.

* 자동차 이름은 쉼표(,)를 기준으로 구분하며 이름은 5자 이하만 가능하다.

* 사용자는 몇 번의 이동을 할 것인지를 입력할 수 있어야 한다.

* 전진하는 조건은 0에서 9사이에서 무작위 값을 구한 후 무작위 값이 4 이상일 경우이다.

* 자동차 경주 게임을 완료한 후 누가 우승했는지를 알려준다. 우승자는 한 명 이상일 수 있다.

* 우승자가 여러 명일 경우 쉼표(,)를 이용하여 구분한다.

* 사용자가 잘못된 값을 입력할 경우 "[ERROR]"로 시작하는 메세지와 함께 `Error`를 발생시킨 후 애플리케이션은 종료되어야 한다.

------

#### 프로그래밍 요구 사항 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를 이용하여 정리한 기능 목록이 정상적으로 작동하는지 테스트 코드로 확인한다.

#### 구현할 기능 목록

1. 경주할 자동차 이름을 입력 받는다. (이름은 쉼표 기준으로 구분한다.)

2. 시도할 횟수를 입력 받는다.

3. 각 경주할 자동차의 0에서 9사이의 무작위 값을 구한 후 무작위 값이 4 이상인 경우, 해당 자동차가 전진 이라고 판단한다.

3-1. 전진한 경우 이름과 - 를 출력한다.

```
pobi : --
woni : ----
jun : --
```

4. 입력받은 시도 횟수 만큼 반복한 뒤, 최종적으로 전진의 횟수가 가장 많은 우승자를 출력한다. `최종 우승자: pobi`

4-1. 우승자는 한 명 이상일 수 있다. 우승자가 여러 명일 경우 쉼표(,)를 이용하여 구분한다.
45 changes: 45 additions & 0 deletions __tests__/ApplicationTest.js
Original file line number Diff line number Diff line change
Expand Up @@ -57,4 +57,49 @@ describe("자동차 경주", () => {
// then
await expect(app.run()).rejects.toThrow("[ERROR]");
});

test("자동차 이름에 공백이 포함된 경우", async () => {
const inputs = ["pobi, ,woni"];
mockQuestions(inputs);

const app = new App();
await expect(app.run()).rejects.toThrow("[ERROR] 이름에 공백이 포함될 수 없습니다.");
});

test("자동차 이름이 5자 이상인 경우", async () => {
const inputs = ["pobi,javaji"];
mockQuestions(inputs);

const app = new App();
await expect(app.run()).rejects.toThrow("[ERROR] 이름은 5자 이하여야 합니다.");
});

test("자동차 이름이 중복된 경우", async () => {
const inputs = ["pobi,woni,pobi"];
mockQuestions(inputs);

const app = new App();
await expect(app.run()).rejects.toThrow("[ERROR] 중복된 이름이 존재합니다.");
});

test("이동 횟수가 숫자가 아닌 경우", async () => {
const inputs = ["pobi,woni", "abc"];
mockQuestions(inputs);

const app = new App();
await expect(app.run()).rejects.toThrow("[ERROR] 시도 횟수는 숫자여야 합니다.");
});

test("이동 횟수가 0 이하인 경우", async () => {
const inputsZero = ["pobi,woni", "0"];
const inputsNegative = ["pobi,woni", "-1"];

mockQuestions(inputsZero);
const app1 = new App();
await expect(app1.run()).rejects.toThrow("[ERROR] 시도 횟수는 1 이상의 숫자로 입력해야 합니다.");

mockQuestions(inputsNegative);
const app2 = new App();
await expect(app2.run()).rejects.toThrow("[ERROR] 시도 횟수는 1 이상의 숫자로 입력해야 합니다.");
});
});
78 changes: 77 additions & 1 deletion src/App.js
Original file line number Diff line number Diff line change
@@ -1,5 +1,81 @@
import { Console } from '@woowacourse/mission-utils';
import { MissionUtils } from '@woowacourse/mission-utils';

class App {
async run() {}
async run() {

const car = await this.Car();
const tryCount = await this.TryCount();

const carCount = new Array(car.length).fill(0);

Console.print('\n실행 결과');
this.startRace(car, carCount, tryCount);

const winner = this.Winner(car, carCount);
Console.print(`최종 우승자 : ${winner.join(", ")}`);
}

async Car() {
Console.print('경주할 자동차 이름을 입력하세요.');
const input = await Console.readLineAsync('');
const car = input.split(",").map(name => name.trim());

this.validCar(input, car);
return car;
}

validCar(input, car) {
// 이름 길이 검사
if (car.some(e => e.length > 5)) {
throw new Error("[ERROR] 이름은 5자 이하여야 합니다.");
}
// 이름 공백 검사
if (car.some(e => e === "")) {
throw new Error("[ERROR] 이름에 공백이 포함될 수 없습니다.");
}
// 중복 검사
if (new Set(car).size !== car.length) {
throw new Error("[ERROR] 중복된 이름이 존재합니다.");
}
}

async TryCount() {
Console.print('시도할 횟수는 몇 회인가요?');
const tryCount = await Console.readLineAsync('');

this.validTryCount(tryCount);
return parseInt(tryCount);
}

validTryCount(tryCount) {
// 숫자 여부
if (isNaN(tryCount) || parseInt(tryCount) != tryCount) {
throw new Error("[ERROR] 시도 횟수는 숫자여야 합니다.");
}
// 1 이상 여부
if (tryCount <= 0) {
throw new Error("[ERROR] 시도 횟수는 1 이상의 숫자로 입력해야 합니다.");
}
}

startRace(car, carCount, tryCount) {
for (let i = 0; i < tryCount; i++) {
for (let j = 0; j < car.length; j++) {
const randomNum = MissionUtils.Random.pickNumberInRange(0, 9);
if (randomNum >= 4) {
carCount[j]++;
}
Console.print(`${car[j]} : ${'-'.repeat(carCount[j])}`);
}
Console.print('\n');
}
}

Winner(car, carCount) {
const WinnerIndex = Math.max(...carCount);
return car.filter((_, index) => carCount[index] === WinnerIndex);
}
}

export default App;