Algorithm/Baekjoon
[백준 2302 : JAVA] 극장 좌석 / DP
팡트루야
2021. 4. 18. 19:04
문제
풀이
점화식을 찾을 수 있는지 묻는 문제이다.
가장 먼저 노가다로 좌석이 4개일 때까지의 경우의 수를 구해보자. (DP에서 점화식을 찾을 때 3~4 정도까지는 노가다로 그려보자)
좌석이 1개일 때)
1
좌석이 2개일 때)
1 2
2 1
좌석이 3개일 때)
1 2 3
1 3 2
2 1 3
좌석이 4개일 때)
1 2 3 4
1 2 4 3
2 1 3 4
2 1 4 3
1 3 2 4
노가다로도 쉽게 구할 수 있는 4개까지 구해보니, [1, 2, 3, 5] 라는 수열을 얻을 수 있었다.
DP[i] = DP[i - 1] + DP[i - 2] 인가... (가장 흔한 피보나치 수열 점화식) 대충 어림짐작해보고 좌석이 5일 경우로 넘어가보자.
좌석이 5개쯤부터는 노가다로 힘들어진다. 이 구간에서는 DP의 철학대로 한 번 쪼개서 생각해보자.
위 경우처럼 쪼개서 구한 경우의 수와 중복되지 않을 경우는 어떤 경우일까?
마지막 5번 좌석이 앞으로 가는 경우다.
더 이상 쪼개는건 무의미하다. 경우의 수에 중복이 발생할 것이기 때문이다.
따라서, 점화식 DP[i] = DP[i - 1] + DP[i - 2] 를 구했다.
코드
import java.io.BufferedReader;
import java.io.InputStreamReader;
public class Main {
private static int N, M;
private static int[] cache;
public static void main(String[] args) throws Exception {
BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
N = Integer.parseInt(br.readLine());
M = Integer.parseInt(br.readLine());
cache = new int[N + 1];
cache[0] = 1;
cache[1] = 1;
cache[2] = 2;
for (int i = 3; i <= N; i++) {
cache[i] = cache[i - 1] + cache[i - 2];
}
int result = 1, prev = 0;
for (int i = 0; i < M; i++) {
int temp = Integer.parseInt(br.readLine());
result *= cache[temp - prev - 1];
prev = temp;
}
result *= cache[N - prev]; // 마지막 vip 좌석 다음 좌석에서 끝 좌석까지의 경우의 수.
System.out.println(result);
}
}