这题勉强算得上是博弈论?(雾
B要胜,就是说要使剩下的石子xor和为0。
计数嘛,数据范围又那么小,很自然地会想到DP。
f[i][j][k]
表示前i堆石子取了余数为j的堆剩下的石子xor和为k的方案数。
显然
f[i][j][k]=f[i−1][j−1][k]+f[i−1][j][k xor a[i]]
。xor状态还没算到?没关系,从小到大排个序就好了。
但是!这样子状态数是
O(vnd)
的,其中v为a中的最大值。显然会爆炸。
首先可以用滚动数组把n这个因子去掉。
但是会发现即使是
2vd
仍然会爆炸。
怎么办呢?
不会搞= =
这是claris的做法:http://www.cnblogs.com/clrs97/p/5006924.html
就是说,根据递推式,
k
和
时间复杂度
O(nlogn+vd)
,空间复杂度
O(vd)
。
#include <bits/stdc++.h>
#define rep(i,a,b) for(int i = a , _ = b ; i <= _ ; i ++)
#define per(i,a,b) for(int i = a , _ = b ; i >= _ ; i --)
#define For(i,a,b) for(int i = a , _ = b ; i < _ ; i ++)
#define Dwn(i,a,b) for(int i = a - 1 , _ = b ; i >= _ ; i --)
inline int rd() {
char c = getchar();
while (!isdigit(c)) c = getchar() ; int x = c - '0';
while (isdigit(c = getchar())) x = x * 10 + c - '0';
return x;
}
const int mod = 1000000007;
inline int add(int a , int b) { a += b ; if (a >= mod) a -= mod ; return a ; }
int a[500001] , tmp[1048576] , f[10][1048576];
int n , d;
void input() {
n = rd() , d = rd();
rep (i , 1 , n) a[i] = rd();
std::sort(a + 1 , a + n + 1);
}
void solve() {
f[0][0] = 1;
int p = 1;
rep (i , 1 , n) {
int t = a[i];
while (p <= t) p <<= 1;
For (i , 0 , p) tmp[i] = add(f[d - 1][i] , f[0][i ^ t]);
Dwn (j , d , 1) For (k , 0 , p) if (k <= (k ^ t)) {
int x = f[j][k];
f[j][k] = add(f[j - 1][k] , f[j][k ^ t]);
f[j][k ^ t] = add(f[j - 1][k ^ t] , x);
}
For (i , 0 , p) f[0][i] = tmp[i];
}
if (n % d == 0) f[0][0] = add(f[0][0] , mod - 1);
printf("%d\n" , f[0][0]);
}
int main() {
#ifndef ONLINE_JUDGE
freopen("data.txt" , "r" , stdin);
#endif
input();
solve();
return 0;
}