洛谷P3185 分裂游戏

时间:2021-11-24 13:05:32

洛谷P3185 分裂游戏

解:这个毒瘤......

我们首先发现每一堆的个数对操作不产生影响,每个操作都是针对单个石子的。

所以等价于每个石子都是一个独立的游戏。把它们异或起来作为sg函数值即可。

单个石子的sg值,直接暴力计算即可。

 #include <bits/stdc++.h>

 const int N = ;

 int sg[N], bin[N], a[N];

 inline int check(int n) {
int ans = ;
for(int i = ; i <= n; i++) {
if(a[i] & ) {
ans ^= sg[n - i];
}
}
return ans;
} inline void solve() {
int n, ans = ;
scanf("%d", &n);
for(int i = ; i <= n; i++) {
scanf("%d", &a[i]);
if(a[i] & ) {
ans ^= sg[n - i];
}
} if(!ans) {
printf("-1 -1 -1\n");
printf("0\n");
return;
} int fd = ;
ans = ; for(int i = ; i < n; i++) {
a[i]--;
for(int j = i + ; j <= n; j++) {
a[j]++;
for(int k = j; k <= n; k++) {
a[k]++;
if(!check(n)) {
ans++;
if(!fd) {
fd = ;
printf("%d %d %d\n", i - , j - , k - );
}
}
a[k]--;
}
a[j]--;
}
a[i]++;
}
printf("%d \n", ans);
return;
} int main() {
sg[] = ;
int n = ;
for(int i = ; i <= n; i++) {
for(int j = ; j < i; j++) {
for(int k = ; k <= j; k++) {
bin[sg[j] ^ sg[k]] = ;
}
}
for(int j = ; ; j++) {
if(!bin[j]) {
sg[i] = j;
break;
}
}
memset(bin, , sizeof(bin));
} /*for(int i = 0; i <= n; i++) {
printf("%d ", sg[i]);
}*/ scanf("%d", &n);
while(n--) {
solve();
} return ;
}

AC代码