코딩/C

[C] 백준 1149번 : RGB거리

Angela_OH 2020. 5. 2. 03:28

 

오늘도 새로운 문제를 풀어보았습니다 !

분명 실버 1 정도의 난이도를 고른 건데,,,,

왜 여태 풀었던 DP 문제들 중 제일 어려웠던 걸까요? ( -᷅_-᷄) 

(결국은 ,, 인터넷의 힘을 빌렸답니다.. ㅎㅎ)

.

백준 1149번

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

 

1149번: RGB거리

첫째 줄에 집의 수 N(2 ≤ N ≤ 1,000)이 주어진다. 둘째 줄부터 N개의 줄에는 각 집을 빨강, 초록, 파랑으로 칠하는 비용이 1번 집부터 한 줄에 하나씩 주어진다. 집을 칠하는 비용은 1,000보다 작거나 같은 자연수이다.

www.acmicpc.net

 

저는 이 문제를 처음 봤을 땐 'dp 배열로 풀면 되겠다!' 라고 생각했습니다.

1번 집부터 N번까지 index를 증가시키면서 dp 배열을 update 해주면 최적화된 값이 나올 것이라 생각했기 때문입니다.

그래서 dp 1차원 배열을 만든 후, dp[집의 개수] = 최소 비용이라고 설정하였습니다.

그 이후 이전 집과 index만 겹치지 않게 dp 배열을 update 해주었습니다.

.

이렇게 코드를 작성하니 아래의 예시에서는 코드가 잘 작동하였습니다.

1149번 예제

 

하지만 제가 임의로 작성해본 예시에 대해서는 틀린 답이 나왔습니다. ㅠ

 

예제 2

 

제가 처음에 작성한 코드는 dp 배열을 1차원으로 만들었기 때문에 현재까지 탐색한 집 정보만 포함합니다.

따라서 집을 계속 탐색하다가 잘못된 값임을 (최솟값이 아님을) 알게 되더라도 이전으로 되돌릴 수 없습니다.

.

예시 1을 보면 1번 집에서 제일 작은 값인 26이 빨간색으로 칠할 때 드는 비용입니다. (첫 번째 index)

따라서 min 비교를 통해 dp[1]에 26을 저장하였습니다. (저는 배열을 1번 index부터 사용하였습니다.)

이렇게 되면 1번 집을 빨간색으로 칠한다고 확정 지어버리는 것입니다.

이 문제는 운이 좋아서 정답이 나왔지만, 예제 2를 보면 이 방식이 틀렸음을 확실히 알 수 있습니다.

.

제가 생각한 방식으로 예제 2를 보면, 1번 집에서는 최솟값인 10 (dp[1] = 10)을 선택합니다.

이후 1번 집을 색칠한 index(index 1)를 제외하고 생각했을 때 최솟값인 것은 1000(index 2)과 또 1000(index 3)입니다.

2번 집에는 9라는 적은 비용을 가진 색깔이 있지만,

dp[1]을 이미 10이라고 선택해버렸기 때문에 되돌아갈 수 없습니다.ㅠ

.

저는 이것을 어떻게 해결하면 좋을까.. 고민하다가 무작정 while 문을 사용해보았습니다.ㅎㅎㅎ

(N 범위 자체가 1000까지이고, 주어진 색깔이 3가지밖에 없어서 어쩌면 풀리지 않을까 싶었습니다...)

저는 모든 case를 다 계산해보고 min을 구해보았습니다.

여기서 반복 횟수를 줄이기 위해, 현재까지 구한 합이 현재 min보다 작으면 다음 반복을 하는 식으로 설정하였습니다.

제가 위에서 언급한 예시 2개와 몇 개의 간단한 예시를 돌려보았을 때는 답이 잘 나왔지만,

역시는 역시.. 시간 초과가 뜨더라구요ㅠㅠ..

(궁금하지 않겠지만..코드가 궁금하다면 더보기를 확인해주세요..ㅎ)

더보기
import java.util.Scanner;

class Main {

	static int N;
	static int[][] input=new int[1001][4];
	static int[] visited=new int[1001];
	static int[] sum=new int[1001];
	static int min=1000*1000+1;
	static int depth=1;
	
	public static void main(String[] args) {
		// TODO Auto-generated method stub
		
		Scanner keyboard=new Scanner(System.in);
		
		N=keyboard.nextInt();
		for (int i=1; i<=N; i++)
			for (int j=1; j<=3; j++)
				input[i][j]=keyboard.nextInt();
		keyboard.close();
		
		outerloop:
		while(true) {
			
			visited[depth]++;
			if (visited[depth-1]==visited[depth])
				visited[depth]++;
			
			while(visited[depth]>3) {
				visited[depth]=0;
				depth--;
				if (depth<=0)
					break outerloop;
				visited[depth]++;
				if (visited[depth-1]==visited[depth])
					visited[depth]++;
			}
		
			sum[depth]=sum[depth-1]+input[depth][visited[depth]];
			if (sum[depth]>=min)
				continue;
			
			depth++;
			if (depth>N) {
				if (sum[depth-1]<min)
					min=sum[depth-1];
				depth--;
			}

		}
		System.out.println(min);
	}

}

 

지금 작성한 코드를 좀 더 수정해볼까 싶었지만

문제 시간제한이 너무 짧기도 하고, 더 최적화할 부분을 찾기가 힘들었습니다.

그래서 인터넷에 다른 분들이 푼 알고리즘을 조금 참고해보았습니다.

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#define MAX 1000*1000

int input[1001][4];
int dp[1001][4];

int min(int x, int y) {
	return x < y ? x : y;
}

int main() {
	int N;
	scanf("%d", &N);
	
	// 입력
	for (int i = 1; i <= N; i++)
		for (int j = 1; j <= 3; j++)
			scanf("%d", &input[i][j]);
	
	// dp 배열 채우기
	dp[1][1] = input[1][1];
	dp[1][2] = input[1][2];
	dp[1][3] = input[1][3];

	for (int i = 2; i <= N; i++) {
		dp[i][1] = min(dp[i - 1][2], dp[i - 1][3]) + input[i][1];
		dp[i][2] = min(dp[i - 1][1], dp[i - 1][3]) + input[i][2];
		dp[i][3] = min(dp[i - 1][1], dp[i - 1][2]) + input[i][3];
	}

	// 출력
	int min = MAX + 1;
	for (int i = 1; i <= 3; i++)
		if (dp[N][i] < min)
			min = dp[N][i];
	printf("%d", min);
	return 0;
}

 

저는 그동안 dp 배열을 1차원으로 잡았었는데, 2차원으로 잡는 방법이 있더라구요...! 나 진짜 바본가ㅠ;

이렇게 되면 현재 구하는 index (여기서는 i)에서 R을 할 때, G를 택할 때, B를 택할 때를 모두 구할 수 있습니다.

그리고 i번째 집의 색깔을 정할 때, i-1번째 집을 확인합니다.

i번째 집 -> 빨간색(index 1)으로 칠하려면

i-1번째 집 -> 초록 색(index 2) 아니면 파란색(index 3)이어야 합니다.

따라서 i-1번째 집을 초록색, 파란색으로 칠했을 때의 비용 중 최솟값에 빨간색을 칠하는 비용을 더하면 됩니다.

=> dp[i][1] = min(dp[i-1][2], dp[i-1][3]) + input[i][1]

즉, 각 집마다 빨간색, 초록 색, 파란색을 칠하는 경우를 모두 고려한 후 조건에 맞게 활용하는 것입니다.

.

위의 예시 2번의 dp 배열을 구해보면 다음과 같습니다.

dp 배열

 

{ }에 적힌 부분은 1번 집부터 선택한 색깔의 index 입니다.

(ex. dp[2][1] -> 1번 집 2번 index (초록 색), 2번 집 1번 index (빨간색) 선택)

이렇게 dp 배열을 채운 후 dp[N][1] ~ dp[N][3] 중 최솟값을 찾으면 됩니다!

.

DP는 정말 왜 이렇게 어려운 걸까요?...^^

앞으로 한참은 더 풀어봐야 할 것 같습니다...