BOJ 28129 - 2022 APC가 어려웠다고요?

image.png

그냥 DP 문제이다.

언뜻보면 시간복잡도가 불가능해보이지만 구간합 배열을 이용하면 $O(N^2)$에 채울 수 있다.

$dp[i][j]=i$ 번째 수가 $j$ 가 되는 경우의 수라고 할 때,

$dp[i][j]=\displaystyle \sum_{k=a_i}^{b_i}dp[i-1][k]$ 일 것이기 때문이다.

const ll mod = 1e9 + 7;
inline ll md(ll x) { return md(mod, x); }
int dp[3001][3001];
void solve() {
   int n, k;
   cin >> n >> k;
   vi a(n + 1), b(n + 1);
   for (int i = 1; i <= n; i++) cin >> a[i] >> b[i];
   for (int i = a[1]; i <= b[1]; i++) dp[1][i] = 1;
   for (int i = 2; i <= n; i++) {
      vi psum(3001);
      for (int j = 1; j <= 3000; j++) psum[j] = md(psum[j - 1] + dp[i - 1][j]);
      auto sum = [&](int l, int r) -> int {
         if (l > r) return 0;
         if (r > 3000) r = 3000;
         return md(psum[r] - (l > 0 ? psum[l - 1] : 0));
      };
      for (int j = a[i]; j <= b[i]; j++) {
         // 현재 수가 j가 되려면
         dp[i][j] = sum(j - k, j + k);
      }
   }
   int ans = 0;
   for (int i = a[n]; i <= b[n]; i++) {
      ans = md(ans + dp[n][i]);
   }
   cout << ans;
}

Tags:

Categories:

Updated:

Comments