알고리즘

BOJ 16985 - Maaaaaaaaaze(Java)

장중앙 2022. 1. 6. 13:31

https://www.acmicpc.net/problem/16985

 

16985번: Maaaaaaaaaze

첫째 줄부터 25줄에 걸쳐 판이 주어진다. 각 판은 5줄에 걸쳐 주어지며 각 줄에는 5개의 숫자가 빈칸을 사이에 두고 주어진다. 0은 참가자가 들어갈 수 없는 칸, 1은 참가자가 들어갈 수 있는 칸을

www.acmicpc.net

 

풀이

1. 2가지를 기준으로 모든 경우의 수를 탐색

  1)  판의 회전(어느 판을 얼만큼 회전할지)

  2)  회전을 완료한 5개의 판 쌓기 

 

2. 위의 기준으로 만든 3차원 행렬을 bfs 거리 탐색

 

 

몇번째 판이 몇도 회전하는지

각 판마다 몇 도 회전하는지 모든 경우의 수를 탐색

    - 현재 판을 90도 회전 시키고 다음 판에 대해 재귀 호출

5번째(마지막)판에 대한 결정이 끝나면 판을 쌓는 함수를 호출 

 private static void searchMinPath(int cnt) {
    if (cnt == 5) {
        // 쌓는순서 정하기
        stackFloor(0);
        return;
    }
    for (int i = 0; i <= 3; i++) {
        //해당 판을 90도 회전하고 다음 판에 대해 재귀 호출
        rotate(cnt);
        searchMinPath(cnt + 1);
    }
}

 

90도 회전

해당 판을 90도 회전

 private static void rotate(int idx) {
    int[][] temp = new int[N][N];
    for (int y = 0; y < N; y++) {
        for (int x = 0; x < N; x++) {
            temp[y][x] = map[idx][N - x - 1][y];
        }
    }
    System.arraycopy(temp, 0, map[idx], 0, temp.length);
}

 

쌓는 순서 정하기

각 판을 몇 번째에 쌓을지 모든 경우의 수를 탐색

5번째(마지막)판에 대한 결정이 끝나면 bfs(거리 탐색)을 호출

 private static void stackFloor(int idx) {
    if (idx == N) {
        bfs();
        return;
    }
    for (int i = 0; i < N; i++) {
        if (visitFloor[i]) continue;
        visitFloor[i] = true;
        floorList.add(i);
        stackFloor(idx + 1);
        floorList.remove(floorList.size() - 1);
        visitFloor[i] = false;
    }
}

 

bfs 도착점까지의 거리 구하기

(0, 0, 0)좌표와 (N-1, N-1, N-1)좌표가 둘다 1(사람이 갈 수 있는 값)인지 확인

(0, 0, 0)에서 시작하여 (N-1, N-1, N-1)까지 몇 번 이동 해야하는지 bfs 탐색

private static void bfs() {
    int[][][] updateMap = new int[N][N][N];
    // 쌓는 순서에 따라 쌓고 bfs
    for (int i = 0; i < N; i++) {
        System.arraycopy(map[floorList.get(i)], 0, updateMap[i], 0, map[floorList.get(i)].length);
    }

    if (updateMap[0][0][0] == 0 || updateMap[4][4][4] == 0) return;

    boolean[][][] visit = new boolean[N][N][N];
    visit[0][0][0] = true;
    Queue<Pos> q = new LinkedList<>();
    q.add(new Pos(0, 0, 0, 0));

    while (!q.isEmpty()) {
        Pos now = q.poll();
        int z = now.z;
        int y = now.y;
        int x = now.x;
        int dist = now.dist;

        if (z == 4 && y == 4 && x == 4) {
            if (MIN > dist) {
                MIN = dist;
            }
            continue;
        }

         for (int d = 0; d < 6; d++) {
            int nz = z + dz[d];
            int ny = y + dy[d];
            int nx = x + dx[d];

            if (nz < 0 || nz >= N || ny < 0 || ny >= N || nx < 0 || nx >= N) continue;
            if (visit[nz][ny][nx] || updateMap[nz][ny][nx] == 0) continue;
            visit[nz][ny][nx] = true;
            q.add(new Pos(nz, ny, nx, dist + 1));
        }
    }
}

전체 코드

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

public class Main {
    static final int INF = 987654321;
    static final int N = 5;
    static int MIN = INF;
    static boolean[] visitFloor = new boolean[N];
    static int[][][] map = new int[N][N][N];
    static List<Integer> floorList = new ArrayList<>();
    static int[] dz = {-1, 1, 0, 0, 0, 0};
    static int[] dy = {0, 0, -1, 1, 0, 0};
    static int[] dx = {0, 0, 0, 0, 1, -1};

    static class Pos {
        int z;
        int y;
        int x;
        int dist;

        Pos(int z, int y, int x, int dist) {
            this.z = z;
            this.y = y;
            this.x = x;
            this.dist = dist;
        }
    }

    public static void main(String[] args) throws IOException {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
//        BufferedReader br = new BufferedReader(new FileReader("a.txt"));

        StringTokenizer st;
        for (int i = 0; i < N; i++) {
            for (int j = 0; j < N; j++) {
                st = new StringTokenizer(br.readLine());
                for (int k = 0; k < N; k++) {
                    map[i][j][k] = Integer.parseInt(st.nextToken());
                }
            }
        }

        searchMinPath(0);

        if (MIN == INF)
            System.out.println(-1);
        else
            System.out.println(MIN);
    }
    // 회전 정하기
    private static void searchMinPath(int cnt) {
        if (cnt == 5) {
            stackFloor(0);
            return;
        }
        for (int i = 0; i <= 3; i++) {
            rotate(cnt);
            searchMinPath(cnt + 1);
        }
    }
    // 쌓는 순서 정하기
    private static void stackFloor(int idx) {
        if (idx == N) {
            bfs();
            return;
        }
        for (int i = 0; i < N; i++) {
            if (visitFloor[i]) continue;
            visitFloor[i] = true;
            floorList.add(i);
            stackFloor(idx + 1);
            floorList.remove(floorList.size() - 1);
            visitFloor[i] = false;
        }
    }

    private static void bfs() {
        int[][][] updateMap = new int[N][N][N];
        // 쌓는 순서에 따라 쌓고 bfs
        for (int i = 0; i < N; i++) {
            System.arraycopy(map[floorList.get(i)], 0, updateMap[i], 0, map[floorList.get(i)].length);
        }

        if (updateMap[0][0][0] == 0 || updateMap[4][4][4] == 0) return;

        boolean[][][] visit = new boolean[N][N][N];
        visit[0][0][0] = true;
        Queue<Pos> q = new LinkedList<>();
        q.add(new Pos(0, 0, 0, 0));

        while (!q.isEmpty()) {
            Pos now = q.poll();
            int z = now.z;
            int y = now.y;
            int x = now.x;
            int dist = now.dist;

            if (z == 4 && y == 4 && x == 4) {
                if (MIN > dist) {
                    MIN = dist;
                }
                continue;
            }

            for (int d = 0; d < 6; d++) {
                int nz = z + dz[d];
                int ny = y + dy[d];
                int nx = x + dx[d];

                if (nz < 0 || nz >= N || ny < 0 || ny >= N || nx < 0 || nx >= N) continue;
                if (visit[nz][ny][nx] || updateMap[nz][ny][nx] == 0) continue;
                visit[nz][ny][nx] = true;
                q.add(new Pos(nz, ny, nx, dist + 1));
            }
        }
    }

    private static void rotate(int idx) {
        int[][] temp = new int[N][N];
        for (int y = 0; y < N; y++) {
            for (int x = 0; x < N; x++) {
                temp[y][x] = map[idx][N - x - 1][y];
            }
        }
        System.arraycopy(temp, 0, map[idx], 0, temp.length);
    }

}