1022: [SHOI2008]小约翰的游戏John


Time Limit: 1 Sec  Memory Limit: 162 MB
Submit: 2976  Solved: 1894
[Submit][Status][Discuss]

Description


  小约翰经常和他的哥哥玩一个非常有趣的游戏:桌子上有n堆石子,小约翰和他的哥哥轮流取石子,每个人取
的时候,可以随意选择一堆石子,在这堆石子中取走任意多的石子,但不能一粒石子也不取,我们规定取到最后一
粒石子的人算输。小约翰相当固执,他坚持认为先取的人有很大的优势,所以他总是先取石子,而他的哥哥就聪明
多了,他从来没有在游戏中犯过错误。小约翰一怒之前请你来做他的参谋。自然,你应该先写一个程序,预测一下
谁将获得游戏的胜利。

Input


  本题的输入由多组数据组成第一行包括一个整数T,表示输入总共有T组数据(T≤500)。每组数据的第一行包
括一个整数N(N≤50),表示共有N堆石子,接下来有N个不超过5000的整数,分别表示每堆石子的数目。

Output


  每组数据的输出占一行,每行输出一个单词。如果约翰能赢得比赛,则输出“John”,否则输出“Brother”
,请注意单词的大小写。

Sample Input


Sample Output


John
Brother

分析:


好吧,我不讲博弈论,讲的太博大精深了。我会尽量讲的通(bo)俗(da)易(jing)懂(shen),

设现在我们所有堆的石头数量都为1。如果有奇数堆,我们是不是输了。。。。设为情况①(必输)

设现在我们所有堆的石头数量都为1。如果有偶数堆,我们是不是赢了。。。。设为情况②(必赢)

(好吧上面两个解释太白痴了。)

设现在我们有一堆石头数量不为2,其他石头都为1.

如果我们有奇数堆数量为1的石头,我们只用把那一堆数量不为2的石头取完,对面就会面临情况①。然后你就赢了2333.

如果我们有偶数堆数量为1的石头,我们只用把那一堆数量不为2的石头变成1,对面就会面临情况①。然后你就赢了2333.

我们把上面两种情况设为情况③(必赢)。

(是不是也很白痴)

好吧,我们再来看另外的情况,我们现在有大于等于2堆石头数量不为2.

如果此时所有石头异或和为奇数我们设为情况④

如果此时所有石头异或和为偶数我们设为情况⑤

如果石头异或和为奇数,我们可以取数量最多的那堆,使异或和变成偶数。此时④可以变成⑤

如果石头异或和为偶数,我们必须取,此时异或和一定变为奇数,⑤又变成了④。然后我们是不是还可以取完一堆石头,也就可能变成③。

所以情况④使对面状态变成⑤,情况⑤会使对面面临状态⑤或③。

所以④(必赢),⑤(必输)。

好了我们就知道了:②③④必赢,①⑤必输,没了。

AC代码:


# include <cstdio>
using namespace std;
int T,n,ans,x,num;
int main(){
scanf("%d",&T);
while(T--){
scanf("%d",&n);
ans = num = ;
for(int i = ;i <= n;i++){
scanf("%d",&x);
if(!x)continue;
ans ^= x;
if(x >= )num++;
}
if((!num && !ans) || (ans && num))puts("John");
else puts("Brother");
}
}
05-11 21:55