不会,抄论文
这是一个非常牛逼的东西,叫做(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;
}