Algorithm/Baekjoon

[백준 15684 : JAVA] 사다리 조작 / 백트래킹

팡트루야 2020. 11. 8. 01:10

문제


 

 

 

풀이


개인적으로 되게 어려웠던 문제였다. 풀이를 보면서도 많은 시간 고민했다.

 

첫 번째로 고민했던건 주어지는 가로선의 정보를 어떤 식으로 저장할까였다. 

그동안 풀었던 문제는 대게 칸으로 주어졌기 때문에 별 생각없이 map[y][x] 에 저장하면 됬지만, y축과 x축이 칸이 아닌 선으로 주어져서 고민했는데, 그냥 아래처럼 칸이라 생각하고 똑같이 저장하면 되겠다라고 생각이 들었다.

 

[그림 1] map 구조

 

위의 그림에서 최상단 좌측을 (1, 1)이라 두고, 값이 저장되는 세 가지 경우를 나눴다.

0 : 해당 칸에는 가로선이 없다.

1 : 해당 칸을 기준으로 우측으로 연결되는 가로선이 있다.

2 : 해당 칸을 기준으로 좌측으로 연결되는 가로선이 있다.

 

1번 높이의 1번과 2번 세로선을 연결하는 가로선이 있다면, map[1][1] = 1; map[1][2] = 2; 이런 식으로 저장했다.

 

그런 다음 i번에서 출발해 i번에 도착하는지 확인하는 체크 용도의 함수가 필요한데, 아래와 같은 코드로 체크했다.

private static boolean check() {
    for (int i = 1; i <= width; i++) {
        int nx = i;
        int ny = 1;

        while (ny <= height) {
            if (map[ny][nx] == 1) nx++; // 우측으로 이동
            else if (map[ny][nx] == 2) nx--; // 좌측으로 이동
            ny++; // y축+1칸 이동한다. (아래로 이동)
        }

        if (nx != i) return false; // i번으로 출발해서 i번으로 도착하지 않는게 하나라도 있다면 리턴.
    }

    return true;
}

 

마지막으로 DFS 탐색의 파라미터로 추가할 가로선의 수를 지정해줘야한다. 이게 DFS 탐색의 종료 조건이 되기 때문이다.

최대 3개의 가로선을 추가할 수 있으므로, 이를 반복문으로 놔두고 각 내부에서 DFS를 호출하는 방법을 이용했다.

 

 

 

코드


import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.StringTokenizer;

public class Main {

    private static BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
    private static int width, height, M, ans;
    private static int[][] map;
    private static boolean isFinish = false;

    public static void main(String[] args) throws IOException {
        StringTokenizer st = new StringTokenizer(br.readLine());
        width = Integer.parseInt(st.nextToken());
        M = Integer.parseInt(st.nextToken()); // 연결된 가로선의 갯수
        height = Integer.parseInt(st.nextToken());

        map = new int[height + 1][width + 1];
        for (int y = 0; y < M; y++) {
            st = new StringTokenizer(br.readLine());

            // a 높이에서 b번과 b+1번 세로선을 연결한다.
            int a = Integer.parseInt(st.nextToken());
            int b = Integer.parseInt(st.nextToken());
            map[a][b] = 1;     // 1 : 우측으로 이동.
            map[a][b + 1] = 2; // 2 : 좌측으로 이동.
        }

        // 추가할 가로선의 갯수를 미리 정해놔야 탐색 종료 조건으로 걸 수 있다.
        // 아래 반복문에서 i는 추가할 가로선의 수.
        for (int i = 0; i <= 3; i++) {
            ans = i;
            dfs(1, 1, 0);
            if (isFinish) break;
        }

        System.out.println((isFinish ? ans : -1));
        br.close();
    }

    // addHorizontalLineNumber : 추가한 가로선의 갯수 (3개가 넘어가면 더이상의 탐색이 무의미.)
    private static void dfs(int x, int y, int addHorizontalLineNumber) {
        if (isFinish) return;
        if (ans == addHorizontalLineNumber) {
            if (check()) isFinish = true;
            return;
        }

        for (int i = y; i <= height; i++) {
            for (int j = x; j < width; j++) {
                // 가로선 두 개가 연속으로 놓여질 수 없기 때문에 가로선을 추가하기 전에 연결된 가로선이 있는지 확인한다.
                if (map[i][j] == 0 && map[i][j + 1] == 0) {
                    // 가로선을 추가한다.
                    map[i][j] = 1;
                    map[i][j + 1] = 2;

                    dfs(1, 1, addHorizontalLineNumber + 1);

                    // 추가했던 가로선을 다시 제거한다. (백트래킹)
                    map[i][j] = 0;
                    map[i][j + 1] = 0;
                }
            }
        }
    }

    // i번으로 출발해서 i번으로 도착하는지 확인한다.
    private static boolean check() {
        for (int i = 1; i <= width; i++) {
            int nx = i;
            int ny = 1;

            while (ny <= height) {
                if (map[ny][nx] == 1) nx++; // 우측으로 이동
                else if (map[ny][nx] == 2) nx--; // 좌측으로 이동
                ny++; // y축+1칸 이동한다. (아래로 이동)
            }

            if (nx != i) return false; // i번으로 출발해서 i번으로 도착하지 않는게 하나라도 있다면 리턴.
        }

        return true;
    }
}