H. Don’t Blame Me

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

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

transition은 좀 특이한데, 모든 비트에 대해 순회하며 $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