게임 개발 문제 (교재 118p)
난이도 ●○○ | 풀이 시간 40분 | 시간 제한 1초 | 메모리 제한 128MB
문제
맵은 N x M 크기의 직사각형이며, 맵의 각 칸은 (A, B)로 나타낼 수 있다.
캐릭터는 상하좌우로 움직일 수 있고, 바다로 되어있는 공간에는 갈 수 없다.
매뉴얼은 아래와 같다.
- 현재 위치에서 현재 방향을 기준으로 왼쪽 방향(반시계 방향으로 90도 회전한 방향)부터 차례대로 갈 곳을 정한다.
- 캐릭터의 바로 왼쪽 방향에 아직 가보지 않은 칸이 존재한다면, 왼쪽 방향을 회전한 다음 왼쪽으로 한 칸을 전진한다. 왼쪽 방향에 가보지 않은 칸이 없다면, 왼쪽 방향으로 회전만 수행하고 1단계로 돌아간다.
- 만약 네 방향 모두 이미 가본 칸이거나 바다로 되어있는 칸의 경우에는, 바라보는 방향을 유지한 채로 한 칸 뒤로 가고 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();
}
}
반응형
'자료구조&알고리즘 > 구현' 카테고리의 다른 글
[이.코.테] Chapter 04 구현 - 왕실의 나이트 (Java) (0) | 2021.04.10 |
---|---|
[이.코.테] Chapter 04 구현 - 시각 (Java) (0) | 2021.04.10 |
[이.코.테] Chapter 04 구현 - 상하좌우 (Java) (0) | 2021.04.10 |
구현 (0) | 2021.04.10 |
댓글