알고리즘
BOJ 16985 - Maaaaaaaaaze(Java)
장중앙
2022. 1. 6. 13:31
https://www.acmicpc.net/problem/16985
풀이
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);
}
}