Skip to content
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
203 changes: 63 additions & 140 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,164 +1,87 @@
# java-racingcar-precourse
# 미션 - 자동차 경주
## 문제 요구사항 분석

## 🔍 진행 방식
주제 : 임의의 갯수의 자동차의 랜덤한 움직임 결과를 출력하는 프로그램

- 미션은 **기능 요구 사항, 프로그래밍 요구 사항, 과제 진행 요구 사항** 세 가지로 구성되어 있다.
- 세 개의 요구 사항을 만족하기 위해 노력한다. 특히 기능을 구현하기 전에 기능 목록을 만들고, 기능 단위로 커밋 하는 방식으로 진행한다.
- 기능 요구 사항에 기재되지 않은 내용은 스스로 판단하여 구현한다.
**자동차**

## 📮 미션 제출 방법
- 이름 조건
- 영어
- 5글자 이하
- 움직임 종류
- 전진
- 멈춤
- 움직임 조건
- 0~9까지의 랜덤 숫자 중 4이상

- 미션 구현을 완료한 후 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점 방지
- 실행 조건
- n개의 자동차 이름 입력
- t번의 움직임 횟수 입력
- 종료 조건
- t번의 움직임이 모두 이루어짐
- 우승 조건
- 가장 많이 움직인 자동차
- 중복 가능

- 기능 구현을 모두 정상적으로 했더라도 **요구 사항에 명시된 출력값 형식을 지키지 않을 경우 0점으로 처리**한다.
- 기능 구현을 완료한 뒤 아래 가이드에 따라 테스트를 실행했을 때 모든 테스트가 성공하는지 확인한다.
- **테스트가 실패할 경우 0점으로 처리**되므로, 반드시 확인 후 제출한다.
**입력**

### 테스트 실행 가이드
- 첫 번째 줄 입력
- 영어
- 5글자 이하
- 쉼표 구분
- 두 번째 줄 입력
- 숫자

- 터미널에서 `java -version`을 실행하여 Java 버전이 17인지 확인한다.
Eclipse 또는 IntelliJ IDEA와 같은 IDE에서 Java 17로 실행되는지 확인한다.
- 터미널에서 Mac 또는 Linux 사용자의 경우 `./gradlew clean test` 명령을 실행하고,
Windows 사용자의 경우 `gradlew.bat clean test` 또는 `./gradlew.bat clean test` 명령을 실행할 때 모든 테스트가 아래와 같이 통과하는지 확인한다.
**출력**

```
BUILD SUCCESSFUL in 0s
```
- 실행 시
- 움직임이 실행 될 때 마다 결과를 출력
- 종료 시
- 가장 많이 움직인 자동차의 이름을 출력
- 중복 시, 쉼표로 구분

---
## 서비스 흐름 구조 기획

## 🚀 기능 요구 사항
<a href="https://ibb.co/TD5WXtHb"><img src="https://i.ibb.co/fV7DRHCn/Untitled-diagram-Mermaid-Chart-2025-09-16-052424-1.png" alt="Untitled-diagram-Mermaid-Chart-2025-09-16-052424-1" border="0"></a>
## 기능 구현 체크리스트

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

- 주어진 횟수 동안 n대의 자동차는 전진 또는 멈출 수 있다.
- 각 자동차에 이름을 부여할 수 있다. 전진하는 자동차를 출력할 때 자동차 이름을 같이 출력한다.
- 자동차 이름은 쉼표(,)를 기준으로 구분하며 이름은 5자 이하만 가능하다.
- 사용자는 몇 번의 이동을 할 것인지를 입력할 수 있어야 한다.
- 전진하는 조건은 0에서 9 사이에서 무작위 값을 구한 후 무작위 값이 4 이상일 경우이다.
- 자동차 경주 게임을 완료한 후 누가 우승했는지를 알려준다. 우승자는 한 명 이상일 수 있다.
- 우승자가 여러 명일 경우 쉼표(,)를 이용하여 구분한다.
- 사용자가 잘못된 값을 입력할 경우 `IllegalArgumentException`을 발생시킨 후 애플리케이션은 종료되어야 한다.
주요 기능 : 입력 값 담당

### 입출력 요구 사항
- [X] 입력 값 받기
- [X] 검증기를 통한 입력 값 검증 과정 추가
- [X] 입력 값 저장

#### 입력
### validator

- 경주 할 자동차 이름(이름은 쉼표(,) 기준으로 구분)
주요 기능 : 입력 값 검증

```
pobi,woni,jun
```
- 이름 검증
- [X] 종류 : 영어
- [X] 글자 수 : 1~5
- [X] 제한 : 구분자 (쉼표)
- 횟수 검증
- [X] 종류 : 숫자

- 시도할 회수
### moveCar

```
5
```
주요 기능 : 자동차의 이동 거리 계산 및 출력

#### 출력
- 이동 거리 계산
- [X] 난수 생성
- [X] 차량 별 이동 거리 저장
- 이동 거리 출력
- [X] 차량 별 이동 거리 출력
- 우승자 출력
- [X] 전체 차량 이동 거리 비교

- 각 차수별 실행 결과
### utils

```
pobi : --
woni : ----
jun : ---
```
주요 기능 : 상수 저장

- 단독 우승자 안내 문구

```
최종 우승자 : pobi
```

- 공동 우승자 안내 문구

```
최종 우승자 : pobi, jun
```

#### 실행 결과 예시

```
경주할 자동차 이름을 입력하세요.(이름은 쉼표(,) 기준으로 구분)
pobi,woni,jun
시도할 회수는 몇회인가요?
5

실행 결과
pobi : -
woni :
jun : -

pobi : --
woni : -
jun : --

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

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

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

최종 우승자 : pobi, jun
```

---

## 🎯 프로그래밍 요구 사항

- JDK 17 버전에서 실행 가능해야 한다. **JDK 17에서 정상적으로 동작하지 않을 경우 0점 처리한다.**
- 프로그램 실행의 시작점은 `Application`의 `main()`이다.
- `build.gradle` 파일을 변경할 수 없고, 외부 라이브러리를 사용하지 않는다.
- [Java 코드 컨벤션](https://github.com/woowacourse/woowacourse-docs/tree/master/styleguide/java) 가이드를 준수하며 프로그래밍한다.
- 프로그램 종료 시 `System.exit()`를 호출하지 않는다.
- 프로그램 구현이 완료되면 `ApplicationTest`의 모든 테스트가 성공해야 한다. **테스트가 실패할 경우 0점 처리한다.**
- 프로그래밍 요구 사항에서 달리 명시하지 않는 한 파일, 패키지 이름을 수정하거나 이동하지 않는다.

### 추가된 요구 사항

- indent(인덴트, 들여쓰기) depth를 3이 넘지 않도록 구현한다. 2까지만 허용한다.
- 예를 들어 while문 안에 if문이 있으면 들여쓰기는 2이다.
- 힌트: indent(인덴트, 들여쓰기) depth를 줄이는 좋은 방법은 함수(또는 메서드)를 분리하면 된다.
- 3항 연산자를 쓰지 않는다.
- 함수(또는 메서드)가 한 가지 일만 하도록 최대한 작게 만들어라.
- JUnit 5와 AssertJ를 이용하여 본인이 정리한 기능 목록이 정상 동작함을 테스트 코드로 확인한다.
- 테스트 도구 사용법이 익숙하지 않다면 `test/java/study`를 참고하여 학습한 후 테스트를 구현한다.

### 라이브러리

- JDK에서 제공하는 Random 및 Scanner API 대신 `camp.nextstep.edu.missionutils`에서 제공하는 `Randoms` 및 `Console` API를 사용하여 구현해야 한다.
- Random 값 추출은 `camp.nextstep.edu.missionutils.Randoms`의 `pickNumberInRange()`를 활용한다.
- 사용자가 입력하는 값은 `camp.nextstep.edu.missionutils.Console`의 `readLine()`을 활용한다.

#### 사용 예시

- 0에서 9까지의 정수 중 한 개의 정수 반환

```java
Randoms.pickNumberInRange(0,9);
```

---

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

- 미션은 [java-racingcar-7](https://github.com/woowacourse-precourse/java-racingcar-7) 저장소를 Fork & Clone해 시작한다.
- **기능을 구현하기 전 `docs/README.md`에 구현할 기능 목록을 정리**해 추가한다.
- **Git의 커밋 단위는 앞 단계에서 `docs/README.md`에 정리한 기능 목록 단위**로 추가한다.
- [커밋 메시지 컨벤션](https://gist.github.com/stephenparish/9941e89d80e2bc58a153) 가이드를 참고해 커밋 메시지를 작성한다.
- 과제 진행 및 제출 방법은 [프리코스 과제 제출](https://github.com/woowacourse/woowacourse-docs/tree/master/precourse) 문서를 참고한다.
- 입력 값 안내 메시지
- 입력 값 오류 메시지
- 정규 표현식
10 changes: 7 additions & 3 deletions src/main/java/racingcar/Application.java
Original file line number Diff line number Diff line change
@@ -1,7 +1,11 @@
package racingcar;

import racingcar.runner.RaceRunner;

public class Application {
public static void main(String[] args) {
// TODO: 프로그램 구현
}

public static void main(String[] args) {
new RaceRunner().run();
}

}
63 changes: 63 additions & 0 deletions src/main/java/racingcar/Application2.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,63 @@
//package racingcar;
//
//import camp.nextstep.edu.missionutils.Console;
//import camp.nextstep.edu.missionutils.Randoms;
//import java.util.HashMap;
//import java.util.List;
//import java.util.Map;
//import java.util.regex.Pattern;
//import racingcar.utils.constant.Constant;
//
//public class Application2 {
//
// public static void main(String[] args) {
// // TODO: 프로그램 구현
//
// // Console 부분
// System.out.println(Constant.REQUIRE_CAR_NAMES);
// String[] cars = Console.readLine().split(",");
// for (String car : cars) {
// if (!Pattern.matches(Constant.REGEX_CAR_NAME, car)) {
// throw new IllegalArgumentException(Constant.CAR_NAME_ERROR);
// }
// }
// System.out.println(Constant.REQUIRE_TRY_NUMBER);
// String tryNumber = Console.readLine();
// if (!Pattern.matches(Constant.REGEX_TRY_NUMBER, tryNumber)) {
// throw new IllegalArgumentException(Constant.TRY_NUMBER_ERROR);
// }
// int tryNumberInt = Integer.parseInt(tryNumber);
//
// //moveCar 부분
// HashMap<String,Integer> carInfo = new HashMap<>();
// for(String car : cars){
// carInfo.put(car,0);
// }
// while(tryNumberInt > 0) {
// for(String car : cars){
// int currentPosition = carInfo.get(car);
// int randomNumber = Randoms.pickNumberInRange(0,9);
// if(randomNumber >=4) {
// currentPosition++;
// carInfo.put(car,currentPosition);
// }
// System.out.println(car + " : " + "-".repeat(currentPosition));
// }
// System.out.println();
// tryNumberInt--;
// }
// // 우승자 출력
// int maxDistance = carInfo.values()
// .stream()
// .max(Integer::compareTo)
// .orElse(0); // 최댓값 구하기
//
// List<String> winners = carInfo.entrySet()
// .stream()
// .filter(entry -> entry.getValue() == maxDistance) // 최댓값과 같은 차량 필터링
// .map(Map.Entry::getKey) // 차량 이름만 추출
// .toList();
// System.out.println("우승자 : " + String.join(",", winners));
// }
//
//}
27 changes: 27 additions & 0 deletions src/main/java/racingcar/controller/RaceController.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
package racingcar.controller;


import camp.nextstep.edu.missionutils.Console;
import racingcar.utils.Validator;
import racingcar.utils.constant.Message;

public class RaceController {

public String[] inputCarNames() {
System.out.println(Message.REQUIRE_CAR_NAMES);
String[] input = Console.readLine().split(",");
String[] cars = new String[input.length];
for(int i = 0 ; i < input.length ; i++) {
cars[i] = input[i].trim();
}
Validator.validateCarName(cars);
return cars;
}

public int inputTryNumber() {
System.out.println(Message.REQUIRE_TRY_NUMBER);
String tryNumber = Console.readLine();
Validator.validateTryNumber(tryNumber);
return Integer.parseInt(tryNumber);
}
}
22 changes: 22 additions & 0 deletions src/main/java/racingcar/domain/Car.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
package racingcar.domain;

public class Car {
private String name;
private int distance;

public Car(String name, int distance) {
this.name = name;
this.distance = distance;
}
public String getName(){
return name;
}

public int getDistance(){
return distance;
}

public void updateDistance(){
distance += 1;
}
}
18 changes: 18 additions & 0 deletions src/main/java/racingcar/domain/Cars.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
package racingcar.domain;

public class Cars {

private Car[] cars;

public Cars(String[] carNames) {
this.cars = new Car[carNames.length];
for (int i = 0; i < carNames.length; i++) {
cars[i] = new Car(carNames[i],0);
}
}

public Car[] getCars(){
return cars;
}

}
Loading