본문 바로가기
알고리즘/백준

[백준 17485] 진우의 달 여행 (Large) (C++)

by fortissimo 2024. 12. 2.

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

문제


우주비행이 꿈이였던 진우는 음식점 '매일매일싱싱'에서 열심히 일한 결과 달 여행에 필요한 자금을 모두 마련하였다! 지구와 우주사이는 N X M 행렬로 나타낼 수 있으며 각 원소의 값은 우주선이 그 공간을 지날 때 소모되는 연료의 양이다.

 

진우는 여행경비를 아끼기 위해 조금 특이한 우주선을 선택하였다. 진우가 선택한 우주선의 특징은 아래와 같다.
1. 지구 -> 달로 가는 경우 우주선이 움직일 수 있는 방향은 아래와 같다.

2. 우주선은 전에 움직인 방향으로 움직일 수 없다. 즉, 같은 방향으로 두번 연속으로 움직일 수 없다.

진우의 목표는 연료를 최대한 아끼며 지구의 어느위치에서든 출발하여 달의 어느위치든 착륙하는 것이다.

최대한 돈을 아끼고 살아서 달에 도착하고 싶은 진우를 위해 달에 도달하기 위해 필요한 연료의 최소값을 계산해 주자.

입력


첫줄에 지구와 달 사이 공간을 나타내는 행렬의 크기를 나타내는 N, M (2 ≤ N, M ≤ 1000)이 주어진다.

다음 N줄 동안 각 행렬의 원소 값이 주어진다. 각 행렬의 원소값은 100 이하의 자연수이다.

 

출력


달 여행에 필요한 최소 연료의 값을 출력한다.

 

문제 풀이


dp 문제.

 

전에 움직인 방향으로 움직일 수 없으므로 이전 칸에서 왼쪽(이동 방향을 나타내는 이미지에서 첫번째)/직진(이미지에서 두번째)/오른쪽(이미지에서 세번째)으로 이동하여 현재 칸에 도달했을 때의 최솟값을 각각 저장한다. 즉 3차원 배열을 이용해 값을 저장한다. 

 

이전 칸(i-1번째 줄)에서 현재 칸(i번째 줄)으로 이동할 때 왼쪽으로 이동했다면, i-2번째 줄에서 i-1번째 줄로 이동했을 때에는 직진 혹은 오른쪽에서 이동했어야 한다. 이전 칸(i-1번째 줄)에서 현재 칸(i번째 줄)으로 이동할 때 직진으로 이동했다면, i-2번째 줄에서 i-1번째 줄로 이동했을 때에는 왼쪽 혹은 오른쪽에서 이동했어야 한다. 이전 칸(i-1번째 줄)에서 현재 칸(i번째 줄)으로 이동할 때 오른쪽으로 이동했다면, i-2번째 줄에서 i-1번째 줄로 이동했을 때에는 왼쪽 혹은 직진으로 이동했어야 한다.

i-1번째 줄에서 i번째 줄으로 이동할 때의 이동방향을 고정해놓으면 i-2번째 줄에서 i-1번째 줄로 이동할 때 가능한 방향이 두 개가 된다. 이 둘 중 작은 값을 선택해 현재 칸에서의 소모 연료의 양을 더해주면 된다.

맨 왼쪽과 맨 오른쪽은 각각 오른쪽 이동 / 왼쪽 이동하여 해당 칸으로 도달할 수 없음에 유의하며 구현해주면 된다.

 

아래는 코드.

더보기
#include <iostream>
#include <algorithm>
#define INF 987654321
using namespace std;

int main()
{
	cin.tie(NULL);
	ios::sync_with_stdio(false);

	int N, M;
	int answer = INF;
	cin >> N >> M;
	int*** dp = new int**[3];
	for (int i = 0; i < 3; i++)
	{
		dp[i] = new int* [N];
		for (int j = 0; j < N; j++)
		{
			dp[i][j] = new int[M];
			for (int k = 0; k < M; k++)
			{
				dp[i][j][k] = INF;
			}
		}
	}
	int** arr = new int* [N];
	for (int i = 0; i < N; i++)
	{
		arr[i] = new int[M];
		for (int j = 0; j < M; j++)
		{
			cin >> arr[i][j];
		}
	}
	//0: 왼쪽, 1: 직진, 2: 오른쪽
	for (int i = 0; i < 3; i++)
	{
		for (int j = 0; j < M; j++)
		{
			dp[i][0][j] = arr[0][j];
		}
	}
	dp[2][0][0] = INF;
	dp[0][0][M - 1] = INF;

	for (int j = 1; j < N; j++)
	{
		dp[0][j][0] = min(dp[1][j - 1][1], dp[2][j-1][1]) + arr[j][0];
		dp[1][j][0] = dp[0][j - 1][0] + arr[j][0];

		for (int k = 1; k < M - 1; k++)
		{
			dp[0][j][k] = min(dp[1][j - 1][k+1], dp[2][j - 1][k+1]) + arr[j][k];
			dp[1][j][k] = min(dp[0][j - 1][k], dp[2][j - 1][k]) + arr[j][k];
			dp[2][j][k] = min(dp[0][j - 1][k-1], dp[1][j - 1][k-1]) + arr[j][k];
		}

		dp[1][j][M - 1] = dp[2][j - 1][M - 1] + arr[j][M - 1];
		dp[2][j][M - 1] = min(dp[0][j - 1][M - 2], dp[1][j-1][M-2]) + arr[j][M - 1];
	}
	for (int i = 0; i < 3; i++)
	{
		for (int j = 0; j < M; j++)
		{
			answer = min(answer, dp[i][N - 1][j]);
		}
	}
	cout << answer << "\n";
	return 0;
}