MapleStory Finger Point

[프로그래머스] 미로 탈출 명령어

2024. 10. 24. 17:39Problem Solving/Online Judge

 

프로그래머스

SW개발자를 위한 평가, 교육, 채용까지 Total Solution을 제공하는 개발자 성장을 위한 베이스캠프

programmers.co.kr

 

💬 문제 설명

n x m 격자 미로가 주어집니다. 당신은 미로의 (x, y)에서 출발해 (r, c)로 이동해서 탈출해야 합니다.

단, 미로를 탈출하는 조건이 세 가지 있습니다.

  1. 격자의 바깥으로는 나갈 수 없습니다.
  2. (x, y)에서 (r, c)까지 이동하는 거리가 총 k여야 합니다. 이때, (x, y)와 (r, c)격자를 포함해, 같은 격자를 두 번 이상 방문해도 됩니다.
  3. 미로에서 탈출한 경로를 문자열로 나타냈을 때, 문자열이 사전 순으로 가장 빠른 경로로 탈출해야 합니다.

이동 경로는 다음과 같이 문자열로 바꿀 수 있습니다.

  • l: 왼쪽으로 한 칸 이동
  • r: 오른쪽으로 한 칸 이동
  • u: 위쪽으로 한 칸 이동
  • d: 아래쪽으로 한 칸 이동

 

예를 들어, 왼쪽으로 한 칸, 위로 한 칸, 왼쪽으로 한 칸 움직였다면, 문자열 "lul"로 나타낼 수 있습니다.

미로에서는 인접한 상, 하, 좌, 우 격자로 한 칸씩 이동할 수 있습니다.

예를 들어 다음과 같이 3 x 4 격자가 있다고 가정해 보겠습니다.

 

....

..S.

E...

 

미로의 좌측 상단은 (1, 1)이고 우측 하단은 (3, 4)입니다. .은 빈 공간, S는 출발 지점, E는 탈출 지점입니다.

탈출까지 이동해야 하는 거리 k가 5라면 다음과 같은 경로로 탈출할 수 있습니다.

 

  1. lldud
  2. ulldd
  3. rdlll
  4. dllrl
  5. dllud
  6. ...

이때 dllrl보다 사전 순으로 빠른 경로로 탈출할 수는 없습니다.

격자의 크기를 뜻하는 정수 n, m, 출발 위치를 뜻하는 정수 x, y, 탈출 지점을 뜻하는 정수 r, c, 탈출까지 이동해야 하는 거리를 뜻하는 정수 k가 매개변수로 주어집니다. 이때, 미로를 탈출하기 위한 경로를 return 하도록 solution 함수를 완성해주세요. 단, 위 조건대로 미로를 탈출할 수 없는 경우 "impossible"을 return 해야 합니다.


제한사항

  • 2 ≤ n (= 미로의 세로 길이) ≤ 50
  • 2 ≤ m (= 미로의 가로 길이) ≤ 50
  • 1 ≤ x ≤ n
  • 1 ≤ y ≤ m
  • 1 ≤ r ≤ n
  • 1 ≤ c ≤ m
  • (x, y) ≠ (r, c)
  • 1 ≤ k ≤ 2,500

입출력 예

n m x y r c k result
3 4 2 3 3 1 5 "dllrl"
2 2 1 1 2 2 2 "dr"
3 3 1 2 3 3 4 "impossible"

입출력 예 설명

입출력 예 #1

문제 예시와 동일합니다.

 

입출력 예 #2

미로의 크기는 2 x 2입니다. 출발 지점은 (1, 1)이고, 탈출 지점은 (2, 2)입니다.

빈 공간은 ., 출발 지점을 S, 탈출 지점을 E로 나타내면 다음과 같습니다.

 

S.

.E

 

미로의 좌측 상단은 (1, 1)이고 우측 하단은 (2, 2)입니다.

탈출까지 이동해야 하는 거리 k가 2이므로 다음과 같은 경로로 탈출할 수 있습니다.

 

  1. rd
  2. dr

"dr"이 사전 순으로 가장 빠른 경로입니다. 따라서 "dr"을 return 해야 합니다.

 

입출력 예 #3

미로의 크기는 3 x 3입니다. 출발 지점은 (1, 2)이고, 탈출 지점은 (3, 3)입니다.

빈 공간은 ., 출발 지점을 S, 탈출 지점을 E로 나타내면 다음과 같습니다.

 

.S.

...

..E

 

미로의 좌측 상단은 (1, 1)이고 우측 하단은 (3, 3)입니다.

탈출까지 이동해야 하는 거리 k가 4입니다. 이때, 이동 거리가 4이면서, S에서 E까지 이동할 수 있는 경로는 존재하지 않습니다.

따라서 "impossible"을 return 해야 합니다.

 

👌 문제 풀이

  1. 시작점과 끝점의 맨해튼 거리를 계산해서 k 횟수 내에 도착할 수 없거나, 최소 경로로 이동하고 남은 이동 횟수가 짝수가 아닐 경우 "impossible"
    • 남은 이동 횟수가 홀수이면 도착 지점으로 다시 돌아올 수 없음(왕복해야 하므로)
  2. 그렇지 않은 경우 최소 경로를 탐색하는 dfs + 모든 경우를 탐색하기 위해 백트래킹
    • 사전순으로 경로를 찾기 위해 d → l → r → u(하 상) 순서대로 이동 
  3. 모든 경우의 수를 다 돌면 비효율적이기 때문에 가지치기 필요
    • 현재 경로의 길이가 k 이상이거나 이미 최적의 경로를 찾았으면 return
    • 남은 이동 횟수로 도착점에 도달할 수 없으면 return

 

💻 코드

import java.util.*;

class Solution {
    
    // 이동 방향: 하, 좌, 우, 상 (사전 순서 고려)
    int[] dy = { 1, 0, 0, -1 };
    int[] dx = { 0, -1, 1, 0 };
    String[] paths = { "d", "l", "r", "u" };
    
    String answer = "impossible"; // 결과 경로 저장
    
    public String solution(int n, int m, int x, int y, int r, int c, int k) {
        // 시작점과 끝점의 맨해튼 거리 계산
        int distance = getDist(x, y, r, c);
        
        // k 이동 횟수 내에 도착할 수 없거나, 남은 이동 횟수가 짝수가 아닐 경우(테스트 31) 불가능
        if (distance > k || (k - distance) % 2 != 0) {
            answer = "impossible";
        } else {
            dfs(n, m, x - 1, y - 1, r - 1, c - 1, k, "");
        }
        
        return answer;
    }
    
    boolean isValid(int n, int m, int y, int x) {
        return 0 <= y && y < n && 0 <= x && x < m;
    }
    
    // 맨해튼 거리 계산
    int getDist(int x1, int y1, int x2, int y2) {
        return Math.abs(y1 - y2) + Math.abs(x1 - x2);
    }
    
    void dfs(int n, int m, int y, int x, int endY, int endX, int k, String path) {
        // 도착점에 도달하고, 경로의 길이가 정확히 k이면 정답 저장
        if (y == endY && x == endX && path.length() == k) {
            answer = path;
            return;
        }

        // 경로 길이가 k보다 크거나 최적 경로를 찾으면 더 이상 탐색할 필요 없음
        if (path.length() >= k || !answer.equals("impossible")) {
            return;
        }

        // 남은 이동 횟수로 도착점에 도달할 수 있는지 확인
        int remainingDistance = getDist(x, y, endX, endY);
        if (remainingDistance > k - path.length()) {
            return;
        }

        // 가능한 모든 방향 탐색 (하, 좌, 우, 상 순서로)
        for (int i = 0; i < 4; i++) {
            int ny = y + dy[i];
            int nx = x + dx[i];

            // 유효한 위치인지 확인
            if (!isValid(n, m, ny, nx)) {
                continue;
            }

            // 백트래킹 탐색 (경로 추가)
            dfs(n, m, ny, nx, endY, endX, k, path + paths[i]);
        }
    }
    
}