Algorithm/Baekjoon

[백준 2225 : JAVA] 합분해 / DP

팡트루야 2021. 4. 8. 23:31

문제


 

 

 

풀이


DP의 핵심 중 하나는 초깃값 설정이다. 

 

해당 문제는 0~N까지의 수열에서 K개의 정수를 선택해 N을 만드는 경우의 수니까 K=1일 경우부터 가정하여 생각해보자.

1개의 수를 선택해 N을 만드는 경우의 수는 무조건 1개이다. (N이라는 수 하나를 선택해야 N을 만들 수 있다.)

 

K=2일 경우를 생각해보자. (N=6 이라 가정)

N=6 이라고 가정하면, [0, 1, 2, 3, 4, 5, 6] 이라는 수열에서 2개의 정수를 선택해 6을 만들기 위한 방법은 아래와 같은 규칙을 가진다.

[0, 1, 2, 3, 4, 5, 6]

[0, 1, 2, 3, 4, 5, 6]

[0, 1, 2, 3, 4, 5, 6]

[0, 1, 2, 3, 4, 5, 6]

 

문제에서 (0, 6)을 선택하는 경우와 (6, 0)을 선택하는 경우는 다르다고 했으니, 2개를 선택해 N을 만드는 경우의 수는 N+1개이다. 

다음으로 K=3일 경우를 생각해보자.

위의 규칙을 이용하면 되는데, 예를 들어 2개를 선택해 6을 만드는 경우 중 하나가 (2, 4)를 선택하는 것이다.

 

그렇다면, 2를 선택했을 때 6을 만들기 위한 수는 4라는 수 하나다. 그럼 2개의 수를 선택해 2라는 수를 만들면, 6을 만들기 위한 나머지 수 하나는 고정되어 있으니, 답을 구할 수 있지 않을까?

// DP[K][N] : K개의 수를 선택해 N이라는 수를 만들 수 있는 경우의 수.
DP[3][6] = DP[2][2];

 

그런데 위와 같은 규칙은 2라는 수뿐만 아니라, 0~N 수 모두에게 적용할 수 있다. 따라서, 다음과 같아진다.

// 2개의 수를 선택해 0을 만들면, 6을 만들기 위해 선택할 수 있는 수는 하나다.
DP[3][6] = DP[2][0] + DP[2][1] + DP[2][2] + ... + DP[2][6];

 

 

 

코드


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

public class Main {

    private static final int MOD = 1_000_000_000;
    private static int N, K;
    private static long[][] cache;

    public static void main(String[] args) throws Exception {
        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
        StringTokenizer st = new StringTokenizer(br.readLine());
        N = Integer.parseInt(st.nextToken());
        K = Integer.parseInt(st.nextToken());
        cache = new long[K + 1][N + 1]; // [정수의 갯수][만들고자하는 수]

        // 1. i개 수를 선택해서 0을 만드는 경우는 무조건 0을 선택하는 한 가지 방법뿐.
        for (int i = 1; i <= K; i++) {
            cache[i][0] = 1;
        }

        // 2. 1개의 수를 선택해 i라는 수를 만들기 위해선 i라는 수를 선택하는 한 가지 방법뿐.
        for (int i = 0; i <= N; i++) {
            cache[1][i] = 1;
        }

        // 3. 2개의 수를 선택해 i라는 수를 만들기 위해선 0~i 까지의 경우의 수를 가진다.
        for (int i = 1; i <= N; i++) {
            cache[2][i] = i + 1;
        }

        // dp[K][N] = dp[K-1][0] + ... + dp[K-1][N]
        for (int i = 3; i <= K; i++) { // 선택할 수 있는 수의 갯수
            for (int j = 1; j <= N; j++) { // 만들고자하는 수
                for (int k = 0; k <= j; k++) {
                    cache[i][j] += (cache[i - 1][k]) % MOD;
                }
            }
        }

        System.out.println(cache[K][N] % MOD);
    }
}