본문 바로가기
자료구조&알고리즘/구현

[이.코.테] Chapter 04 구현 - 게임 개발 (Java)

by _din 2021. 4. 10.

게임 개발 문제 (교재 118p)

난이도 ●○○ | 풀이 시간 40분 | 시간 제한 1초 | 메모리 제한 128MB

 

문제

맵은 N x M 크기의 직사각형이며, 맵의 각 칸은 (A, B)로 나타낼 수 있다.

캐릭터는 상하좌우로 움직일 수 있고, 바다로 되어있는 공간에는 갈 수 없다.

매뉴얼은 아래와 같다.

  1. 현재 위치에서 현재 방향을 기준으로 왼쪽 방향(반시계 방향으로 90도 회전한 방향)부터 차례대로 갈 곳을 정한다.
  2. 캐릭터의 바로 왼쪽 방향에 아직 가보지 않은 칸이 존재한다면, 왼쪽 방향을 회전한 다음 왼쪽으로 한 칸을 전진한다. 왼쪽 방향에 가보지 않은 칸이 없다면, 왼쪽 방향으로 회전만 수행하고 1단계로 돌아간다.
  3. 만약 네 방향 모두 이미 가본 칸이거나 바다로 되어있는 칸의 경우에는, 바라보는 방향을 유지한 채로 한 칸 뒤로 가고 1단계로 돌아간다. 단, 이때 뒤쪽 방향이 바다인 칸이라 뒤로 갈 수 없는 경우에는 움직임을 멈춘다.

캐릭터가 방문한 칸의 출력하는 프로그램을 만드시오.

 

입출력 조건)

입력 조건  - 첫째 줄에 맵의 세로 크기 N과 가로크기 M을 공백으로 구분하여 입력한다. (3 ≤ N, M ≤ 50)
 - 둘째 줄에 게임 캐릭터가 있는 칸의 좌표 (A, B)와 바라보는 방향 d가 각각 서로 공백으로 구분하여 주어진다. 방향 d의 값으로는 다음과 같이 4가지가 존재한다.
  - 0 : 북쪽
  - 1 : 동쪽
  - 2 : 남쪽
  - 3 : 서쪽
 - 셋재 줄부터 맵이 육지인지 바다인지에 대한 정보가 주어진다. N개의 줄에 맵의 상태가 북쪽부터 남쪽 순서대로, 각 줄의 데이터는 서쪽부터 동쪽 순서대로 주어진다. 맵의 외곽은 항상 바다로 되어있다.
  - 0 : 육지
  - 1 : 바다
 - 처음에 게임 캐릭터가 위치한 칸의 상태는 항상 육지이다.
출력 조건  첫째 줄에 이동을 마친 후 캐릭터가 방문한 칸의 수를 출력한다.

입출력 예시)

입력 예시 출력 예시
4 4
1 1 0
1 1 1 1
1 0 0 1
1 1 0 1
1 1 1 1
3

문제 풀이 방법

  • 일반적으로 방향을 설정해서 이동하는 문제 유형에서는 dx, dy라는 별도의 리스트를 만들어 방향을 정하는 것이 효과적이다.
  • 왼쪽으로 회전하는 함수 turn_left() 구현

 

실수한 점 & 부족한 점

  • 코드는 비슷하게 짰는데, if문 조건을 세우는 데 미흡했다.
    • 네 방향 모두 갈 수없는 경우, if문 안에 &&을 써서 좀 복잡하게 작성했었다.
  • 코드의 순서를 세우는 데 어려움을 겪었다.
    • 네 방향 모두 갈 수 없는 경우를 가장 위쪽에 위치해두었다.
  • 책에서는 아래와 같이 1차원 배열로 x방향 y방향 나누어 선언하였고,
public static int[] dx = {-1, 0, 1, 0}; // 북, 동, 남, 서
public static int[] dy = {0, 1, 0, -1};

 

      나는 아래와 같이 2차원 배열의 방향으로 작성하여 코드가 가독성이 떨어지고 복잡하였다.

public static int[][] dir = {{-1,0},{0,1},{1,0},{0,-1}}; // 북동남서

소스 코드

import java.io.*;
import java.util.StringTokenizer;

public class Q_DevelopGame {
    public static int[] dx = {-1, 0, 1, 0}; // 북, 동, 남, 서
    public static int[] dy = {0, 1, 0, -1};
    public static int direction;

    public static void turn_left(){
        direction -= 1;
        if(direction == -1)
            direction = 3;
    }

    public static int solution(int n, int m, int a, int b, int d, int[][] map) {
        int count = 1; // 움직인 횟수
        int turn_time = 0; // 회전 수
        int curX = a; // 현재 x 좌표
        int curY = b; // 현재 y 좌표
        int nextX = 0; // 다음 x 좌표
        int nextY = 0; // 다음 y 좌표
        boolean[][] visitied = new boolean[n][m]; // 방문 여부 확인
        visitied[a][b] = true; // 현재 좌표 방문 처리
        direction = d;

        while(true){
            // 왼쪽으로 회전
            turn_left();

            nextX = curX + dx[direction];
            nextY = curY + dy[direction];

            // 회전한 이후 정면에 가보지 않았으며 육지인 칸이 존재하는 경우 이동
            if(!visitied[nextX][nextY] && map[nextX][nextY] == 0) {
                visitied[nextX][nextY] = true; // 방문 처리
                curX = nextX; // 현재 x 위치 변경
                curY = nextY; // 현재 y 위치 변경
                count += 1; // 방문횟수 더해주기
                turn_time = 0; // 회전 수 초기화
            } else{
                turn_time += 1;
            }

            // 네 방향 모두 갈 수 없는 경우
            if(turn_time == 4){
                nextX = curX - dx[direction];
                nextY = curY - dy[direction];

                // 뒤로 갈 수 있다면 이동하기
                if(map[nextX][nextY] == 0){
                    curX = nextX;
                    curY = nextY;
                } else { // 뒤가 바다로 막혀있는 경우
                    break;
                }
                turn_time = 0; // 회전 수 초기화
            }

        }

        return count;

    }

    public static void main(String[] args) throws IOException {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(System.out));

        StringTokenizer st = new StringTokenizer(br.readLine());
        int n = Integer.parseInt(st.nextToken());
        int m = Integer.parseInt(st.nextToken());
        st = new StringTokenizer(br.readLine());

        int a = Integer.parseInt(st.nextToken());
        int b = Integer.parseInt(st.nextToken());
        int d = Integer.parseInt(st.nextToken());

        int[][] map = new int[n][m];
        for (int row = 0; row < n; row++) {
            st = new StringTokenizer(br.readLine());
            for (int col = 0; col < m; col++) {
                map[row][col] = Integer.parseInt(st.nextToken());
            }
        }

        System.out.println(solution(n,m,a,b,d,map));

        br.close();
        bw.close();
    }
}

 

 

반응형

댓글