알고리즘/BAEKJOON

BOJ 6593 : 상범 빌딩

피로물든딸기 2023. 3. 26. 15:46
반응형

알고리즘 문제 전체 링크

 

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

 

참고

- BOJ 11779 : 최소비용 구하기 2

 

 

최단 경로는 BOJ 11779 : 최소비용 구하기 2에서 사용한 다익스트라를 이용해서 구할 수 있다.

상범 빌딩은 다익스트라를 3차원에 적용하면 된다.

 

tc가 여러 개이므로 매 tc마다 dist를 INF로 visit을 0으로 초기화한다.

	for (i = 1; i <= L; i++)
		for (k = 1; k <= R; k++)
			for (t = 1; t <= C; t++)
				dist[i][k][t] = INF;

	for (i = 1; i <= L; i++)
		for (k = 1; k <= R; k++)
			for (t = 1; t <= C; t++)
				visit[i][k][t] = 0;

 

다익스트라를 적용하기 위해 2차원 탐색을 위한 dr, dc 배열과 3차원 탐색을 위한 dl 배열을 선언한다.

int dr[] = { 0, 1, 0, -1 };
int dc[] = { 1, 0, -1, 0 };
int dl[] = { -1, 1 };

 

아래와 같이 현재 좌표를 기준으로 2차원 네 방향과, 3차원 두 방향에 대해 다익스트라를 실행하면 된다.

	while (hn)
	{
		HEAP tmp;

		tmp = pop();

		if (visit[tmp.l][tmp.r][tmp.c]) continue;
		visit[tmp.l][tmp.r][tmp.c] = 1;

		for (i = 0; i < 4; i++)
		{
			nr = tmp.r + dr[i];
			nc = tmp.c + dc[i];

			if (nr < 1 || nc < 1 || nr > R || nc > C) continue;
			if (MAP[tmp.l][nr][nc] == '#') continue;
			if (dist[tmp.l][nr][nc] > dist[tmp.l][tmp.r][tmp.c] + 1)
			{
				dist[tmp.l][nr][nc] = dist[tmp.l][tmp.r][tmp.c] + 1;
				push({ tmp.l, nr, nc, dist[tmp.l][nr][nc] });
			}
		}

		for (i = 0; i < 2; i++)
		{
			nl = tmp.l + dl[i];

			if (nl < 1 || nl > L) continue;
			if (MAP[nl][tmp.r][tmp.c] == '#') continue;
			if (dist[nl][tmp.r][tmp.c] > dist[tmp.l][tmp.r][tmp.c] + 1)
			{
				dist[nl][tmp.r][tmp.c] = dist[tmp.l][tmp.r][tmp.c] + 1;
				push({ nl, tmp.r, tmp.c, dist[nl][tmp.r][tmp.c] });
			}
		}
	}

 

전체 코드는 다음과 같다.

#include <stdio.h>

#define INF (0x7fff0000)

int L, R, C;
char MAP[35][35][35];

typedef struct st
{
	int l;
	int r;
	int c;
	int value;
}HEAP;

HEAP heap[35 * 35 * 35 * 10];
int hn;

HEAP pop()
{
	int i;
	HEAP ret, tmp;

	ret = heap[1];
	heap[1] = heap[hn];
	heap[hn--].value = INF;

	for (i = 1; i * 2 <= hn;)
	{
		if (heap[i].value < heap[i * 2].value && heap[i].value < heap[i * 2 + 1].value) break;
		else if (heap[i * 2].value < heap[i * 2 + 1].value)
		{
			tmp = heap[i * 2];
			heap[i * 2] = heap[i];
			heap[i] = tmp;

			i = i * 2;
		}
		else
		{
			tmp = heap[i * 2 + 1];
			heap[i * 2 + 1] = heap[i];
			heap[i] = tmp;

			i = i * 2 + 1;
		}
	}

	return ret;
}

void push(HEAP x)
{
	int i;
	HEAP tmp;

	heap[++hn] = x;

	for (i = hn; i > 1; i /= 2)
	{
		if (heap[i / 2].value < heap[i].value) break;

		tmp = heap[i];
		heap[i] = heap[i / 2];
		heap[i / 2] = tmp;
	}
}

int dist[35][35][35];
int visit[35][35][35];
int dr[] = { 0, 1, 0, -1 };
int dc[] = { 1, 0, -1, 0 };
int dl[] = { -1, 1 };

int main()
{
	int i, k, t, nl, nr, nc;
	int sr, sc, sl, er, ec, el;

	while (1)
	{
		scanf("%d %d %d", &L, &R, &C);
		if (!(L + R + C)) break;
		for (i = 1; i <= L; i++)
		{
			for (k = 1; k <= R; k++)
			{
				for (t = 1; t <= C; t++)
				{
					scanf(" %c", &MAP[i][k][t]);
					if (MAP[i][k][t] == 'S') sl = i, sr = k, sc = t;
					else if (MAP[i][k][t] == 'E') el = i, er = k, ec = t;
				}
			}
		}

		for (i = 1; i <= L; i++)
			for (k = 1; k <= R; k++)
				for (t = 1; t <= C; t++)
					dist[i][k][t] = INF;

		for (i = 1; i <= L; i++)
			for (k = 1; k <= R; k++)
				for (t = 1; t <= C; t++)
					visit[i][k][t] = 0;

		dist[sl][sr][sc] = 0;

		push({ sl, sr ,sc, 0 });

		while (hn)
		{
			HEAP tmp;

			tmp = pop();

			if (visit[tmp.l][tmp.r][tmp.c]) continue;
			visit[tmp.l][tmp.r][tmp.c] = 1;

			for (i = 0; i < 4; i++)
			{
				nr = tmp.r + dr[i];
				nc = tmp.c + dc[i];

				if (nr < 1 || nc < 1 || nr > R || nc > C) continue;
				if (MAP[tmp.l][nr][nc] == '#') continue;
				if (dist[tmp.l][nr][nc] > dist[tmp.l][tmp.r][tmp.c] + 1)
				{
					dist[tmp.l][nr][nc] = dist[tmp.l][tmp.r][tmp.c] + 1;
					push({ tmp.l, nr, nc, dist[tmp.l][nr][nc] });
				}
			}

			for (i = 0; i < 2; i++)
			{
				nl = tmp.l + dl[i];

				if (nl < 1 || nl > L) continue;
				if (MAP[nl][tmp.r][tmp.c] == '#') continue;
				if (dist[nl][tmp.r][tmp.c] > dist[tmp.l][tmp.r][tmp.c] + 1)
				{
					dist[nl][tmp.r][tmp.c] = dist[tmp.l][tmp.r][tmp.c] + 1;
					push({ nl, tmp.r, tmp.c, dist[nl][tmp.r][tmp.c] });
				}
			}
		}

		if (dist[el][er][ec] == INF) printf("Trapped!\n");
		else printf("Escaped in %d minute(s).\n", dist[el][er][ec]);
	}

	return 0;
}
반응형