H. Don’t Blame Me

dp[i][b]=idp[i][b]=i 번째 수 까지 봤을 때 수들을 AND\text{AND} 해서 bb가 나오는 경우의 수라고 하면

이걸 O(64N)O(64 \cdot N) 에 구할 수 있다.

transition은 좀 특이한데, 모든 비트에 대해 순회하며 dp[i][ai & bit]dp[i][ai & bit]+dp[i1][bit]dp[i][a_i ~\&~ \text{bit}] \coloneqq dp[i][a_i ~\&~ \text{bit}]+dp[i-1][\text{bit}] 처럼 해줄 수 있다.

const ll mod = 1e9 + 7;  
inline ll md(ll x) { return md(mod, x); }  
void solve() {  
   int n, k;  
   cin >> n >> k;  
   vi a(n + 1);  
   fv1(a);  
  
   // i 번째 수까지 봤을 때 and 해서 해당 숫자가 나오는 경우의 수  
   vi dp(64, 0);  
   for (int i = 1; i <= n; i++) {  
      vi tmp(64);  
      tmp[a[i]] = 1;  
      for (int b = 0; b < 64; b++) {  
         tmp[b & a[i]] += dp[b];  
         tmp[b & a[i]] %= mod;  
      }  
      for (int b = 0; b < 64; b++) dp[b] = md(dp[b] + tmp[b]);  
   }  
   int ans = 0;  
   for (int mask = 0; mask < 64; mask++) {  
      if (__builtin_popcount(mask) == k) {  
         ans = md(ans + dp[mask]);  
      }  
   }  
   cout << ans << endl;  
}

Comments