[SHOI2008]小约翰的游戏
不会,抄论文
这是一个非常牛逼的东西,叫做\(anti\)博弈,就是进行最后一次操作的人输
我们考虑一下这道题
显然如果石子个数都是\(1\),那么有奇数堆石子先手必败,有偶数堆石子先手必胜
如果只有一堆石子大于\(1\),如果当前是奇数堆石子,那么我们可以把这堆石子取到只剩下一个,如果是偶数堆石子,我们直接把大于\(1\)的这一堆都拿走,这样我们就能让对方面对必败局面
考虑刚才的局面的\(SG\)函数值,显然不为\(0\)
而我们发现随着游戏的进行石子的减少,刚才的局面一定会出现
所以只要\(SG\)函数不为\(0\)且不是最开始的那种全是\(1\)的情况就能判定先手必胜了
因此我们得到了针对\(anti-SG\)的\(SJ\)定理
对于一个\(anti-SG\)游戏先手必胜的条件为满足下两个条件之一
-
\(SG\)函数大于\(0\),且有至少一个单个游戏的\(SG\)函数大于\(1\)
-
\(SG\)函数等于\(0\),且没有任何一个单个游戏的\(SG\)函数大于\(1\)
代码
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#define re register
#define LL long long
#define max(a,b) ((a)>(b)?(a):(b))
#define min(a,b) ((a)<(b)?(a):(b))
inline int read() {
char c=getchar();int x=0;while(c<'0'||c>'9') c=getchar();
while(c>='0'&&c<='9') x=(x<<3)+(x<<1)+c-48,c=getchar();return x;
}
int T,a,n;
int main() {
T=read();
while(T--) {
n=read();int tot=0,s=0;
for(re int i=1;i<=n;i++)
a=read(),tot+=(a==1),s^=a;
if(tot==n) puts(s?"Brother":"John");
else puts(s?"John":"Brother");
}
return 0;
}
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#define re register
#define LL long long
#define max(a,b) ((a)>(b)?(a):(b))
#define min(a,b) ((a)<(b)?(a):(b))
inline int read() {
char c=getchar();int x=0;while(c<'0'||c>'9') c=getchar();
while(c>='0'&&c<='9') x=(x<<3)+(x<<1)+c-48,c=getchar();return x;
}
int T,a,n;
int main() {
T=read();
while(T--) {
n=read();int tot=0,s=0;
for(re int i=1;i<=n;i++)
a=read(),tot+=(a==1),s^=a;
if(tot==n) puts(s?"Brother":"John");
else puts(s?"John":"Brother");
}
return 0;
}