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
80 changes: 79 additions & 1 deletion README.md
Original file line number Diff line number Diff line change
@@ -1 +1,79 @@
# java-racingcar-precourse
# java-racingcar-precourse

# 기능 요구 사항

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

- 주어진 횟수 동안 n대의 자동차는 전진 또는 멈출 수 있다.
- 각 자동차에 이름을 부여할 수 있다. 전진하는 자동차를 출력할 때 자동차 이름을 같이 출력한다.
- 자동차 이름은 쉼표(,)를 기준으로 구분하며 이름은 5자 이하만 가능하다.
- 사용자는 몇 번의 이동을 할 것인지를 입력할 수 있어야 한다.
- 전진하는 조건은 0에서 9 사이에서 무작위 값을 구한 후 무작위 값이 4 이상일 경우이다.
- 자동차 경주 게임을 완료한 후 누가 우승했는지를 알려준다. 우승자는 한 명 이상일 수 있다.
- 우승자가 여러 명일 경우 쉼표(,)를 이용하여 구분한다.
- 사용자가 잘못된 값을 입력할 경우 `IllegalArgumentException`를 발생시키고, "[ERROR]"로 시작하는 에러 메시지를 출력 후 그 부분부터 입력을 다시 받는다.
- `Exception`이 아닌 `IllegalArgumentException`, `IllegalStateException` 등과 같은 명확한 유형을 처리한다.

## 프로그래밍 요구 사항 1
- JDK 17버전에서 실행 가능해야 한다.
- 프로그램 실행의 시작점은 Application 의 main() 이다.
- build.gradle 파일은 변경할 수 없으며, 제공된 라이브러리 이외의 외부 라이브러리는 사용하지 않는다.
- 프로그램 종료 시 System.exit() 를 호출하지 않는다.
- 프로그래밍 요구 사항에서 달리 명시하지 않는 한 파일, 패키지 등의 이름을 바꾸거나 이동하지 않는다.

## 프로그래밍 요구 사항 2
- 자바 코드 컨벤션을 지키면서 프로그래밍한다.
- 기본적으로 Google Java Style Guide을 원칙으로 한다.
- 단, 들여쓰기는 '2 spaces'가 아닌 '4 spaces'로 한다.
- indent(인덴트, 들여쓰기) depth를 3이 넘지 않도록 구현한다. 2까지만 허용한다.
- 예를 들어 while문 안에 if문이 있으면 들여쓰기는 2이다.
- 힌트: indent(인덴트, 들여쓰기) depth를 줄이는 좋은 방법은 함수(또는 메서드)를 분리하면 된다.
- 3항 연산자를 쓰지 않는다.
- 함수(또는 메서드)가 한 가지 일만 하도록 최대한 작게 만들어라.
- JUnit 5와 AssertJ를 이용하여 정리한 기능 목록이 정상적으로 작동하는지 테스트 코드로 확인한다.
- 테스트 도구 사용법이 익숙하지 않다면 아래 문서를 참고하여 학습한 후 테스트를 구현한다.
- JUnit 5 User Guide
- AssertJ User Guide
- AssertJ Exception Assertions
- Guide to JUnit 5 Parameterized Tests

## 프로그래밍 요구 사항 3
- 함수(또는 메서드)의 길이가 15라인을 넘어가지 않도록 구현한다.
- 함수(또는 메서드)가 한 가지 일만 잘 하도록 구현한다.
- else 예약어를 쓰지 않는다.
- else를 쓰지 말라고 하니 switch/case로 구현하는 경우가 있는데 switch/case도 허용하지 않는다.
- 힌트: if 조건절에서 값을 return하는 방식으로 구현하면 else를 사용하지 않아도 된다.
- 도메인 로직에 단위 테스트를 구현해야 한다. 단, UI(System.out, System.in, Scanner) 로직은 제외한다.
- 핵심 로직을 구현하는 코드와 UI를 담당하는 로직을 분리해 구현한다.
- 힌트: MVC 패턴 기반으로 구현한 후, View와 Controller를 제외한 Model에 대한 단위 테스트 추가에 집중한다.

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

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

pobi : --
woni : -
jun : --

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

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

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

최종 우승자 : pobi, jun
```
Empty file removed src/main/java/.gitkeep
Empty file.
8 changes: 8 additions & 0 deletions src/main/java/Application.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
import controller.RacingGameController;

public class Application {
public static void main(String[] args) {
RacingGameController controller = new RacingGameController();
controller.start();
}
}
27 changes: 27 additions & 0 deletions src/main/java/controller/RacingGameController.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
package controller;

import model.RacingGame;
import view.View;

import java.util.List;

public class RacingGameController {
private final View view = new View();

/**
* 게임을 시작하는 메서드
*/
public void start() {
List<String> carNames = view.getCarNames();
int trialCount = view.getTrialCount();
RacingGame racingGame = new RacingGame(carNames);

for (int i = 0; i < trialCount; i++) {
racingGame.race(1);
view.displayRaceResult(racingGame.getCars());
System.out.println();
}

view.displayWinners(racingGame.getWinners());
}
}
66 changes: 66 additions & 0 deletions src/main/java/model/Car.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,66 @@
package model;

public class Car {
private final String name;
private int position;
private Integer fixedRandomNumber = null;

/**
* 자동차 이름을 받아 생성하는 생성자
* @param name 자동차 이름
*/
public Car(String name) {
if (name.length() > 5) {
throw new IllegalArgumentException("자동차 이름은 5자 이하만 가능합니다.");
}
this.name = name;
this.position = 0;
}

/**
* 테스트용으로 랜덤 숫자를 고정하는 메서드
* @param number 고정할 랜덤 숫자
*/
public void setFixedRandomNumber(int number) {
this.fixedRandomNumber = number;
}

/**
* 랜덤 숫자를 생성하여 이동하는 메서드
* 랜덤 숫자가 4 이상일 때 이동
*/
public void move() {
if (getRandomNumber() >= 4) {
this.position++;
}
}

/**
* 랜덤 숫자를 생성하는 메서드
* 테스트용으로 랜덤 숫자를 고정했을 경우 고정된 숫자를 반환
* @return 생성된 랜덤 숫자
* @see Car#move()
*/
protected int getRandomNumber() {
if (fixedRandomNumber != null) {
return fixedRandomNumber;
}
return (int) (Math.random() * 10);
}

/**
* 현재 자동차의 이름을 반환하는 메서드
* @return 자동차 이름
*/
public String getName() {
return name;
}

/**
* 현재 자동차의 위치를 반환하는 메서드
* @return 자동차 위치
*/
public int getPosition() {
return position;
}
}
56 changes: 56 additions & 0 deletions src/main/java/model/RacingGame.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,56 @@
package model;

import java.util.ArrayList;
import java.util.List;
import java.util.stream.Collectors;

public class RacingGame {
private final List<Car> cars;

/**
* 자동차 이름 리스트를 받아 생성하는 생성자
* @param carNames 자동차 이름 리스트
*/
public RacingGame(List<String> carNames) {
cars = new ArrayList<>();
for (String name : carNames) {
cars.add(new Car(name));
}
}

/**
* 주어진 횟수만큼 자동차 경주를 진행하는 메서드
* @param trials 시도할 횟수
*/
public void race(int trials) {
for (int i = 0; i < trials; i++) {
for (Car car : cars) {
car.move();
}
}
}

/**
* 현재 자동차 리스트를 반환하는 메서드
* @return 현재 자동차 리스트
*/
public List<Car> getCars() {
return cars;
}

/**
* 현재 가장 멀리 있는 자동차의 위치를 반환하는 메서드
* @return 현재 가장 멀리 있는 자동차의 위치
*/
public List<String> getWinners() {
int maxPosition = cars.stream()
.mapToInt(Car::getPosition)
.max()
.orElse(0);

return cars.stream()
.filter(car -> car.getPosition() == maxPosition)
.map(Car::getName)
.collect(Collectors.toList());
}
}
55 changes: 55 additions & 0 deletions src/main/java/view/View.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,55 @@
package view;

import model.Car;

import java.util.List;
import java.util.Scanner;

public class View {
private final Scanner scanner = new Scanner(System.in);

/**
* 사용자로부터 자동차 이름을 입력받는 메서드
* @return 입력받은 자동차 이름 리스트
*/
public List<String> getCarNames() {
while (true) {
try {
System.out.println("경주할 자동차 이름을 입력하세요.(이름은 쉼표(,) 기준으로 구분)");
String input = scanner.nextLine();
String[] names = input.split(",");
for (String name : names) {
new Car(name); // 이 과정에서 예외가 발생할 수 있음
}
return List.of(names);
} catch (IllegalArgumentException e) {
System.out.println("[ERROR] " + e.getMessage());
}
}
}

/**
* 사용자로부터 시도할 회수를 입력받는 메서드
* @return 입력받은 시도할 회수
*/
public int getTrialCount() {
while (true) {
try {
System.out.println("시도할 회수는 몇회인가요?");
return Integer.parseInt(scanner.nextLine());
} catch (NumberFormatException e) {
System.out.println("[ERROR] 유효한 숫자를 입력하세요.");
}
}
}

public void displayRaceResult(List<Car> cars) {
for (Car car : cars) {
System.out.println(car.getName() + " : " + "-".repeat(car.getPosition()));
}
}

public void displayWinners(List<String> winners) {
System.out.println("최종 우승자 : " + String.join(", ", winners));
}
}
44 changes: 44 additions & 0 deletions src/test/java/CarTest.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
import model.Car;
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.ValueSource;

import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertThrows;

public class CarTest {

@ParameterizedTest
@ValueSource(ints = {4, 5, 6, 7, 8, 9})
@DisplayName("랜덤 숫자가 4 이상일 때 이동 검증")
void move(int randomNumber) {
Car car = new Car("Test");
car.setFixedRandomNumber(randomNumber);
car.move();
assertEquals(1, car.getPosition());
}

@ParameterizedTest
@ValueSource(ints = {0, 1, 2, 3})
@DisplayName("랜덤 숫자가 4 미만일 때 움직이지 않는 것 검증")
void stop(int randomNumber) {
Car car = new Car("Test");
car.setFixedRandomNumber(randomNumber);
car.move();
assertEquals(0, car.getPosition());
}

@ParameterizedTest
@ValueSource(strings = {"abcdef", "123456", "longname"})
@DisplayName("자동차 이름이 5자를 초과하면 예외 발생 검증")
void nameEx(String name) {
assertThrows(IllegalArgumentException.class, () -> new Car(name));
}

@ParameterizedTest
@ValueSource(strings = {"abc", "12345", "long"})
@DisplayName("자동차 이름이 5자 이하면 예외가 없음")
void nameOk(String name) {
new Car(name); // 예외가 발생하지 않으면 테스트 통과
}
}
Loading