标签:sample 比赛 优势 -- 组成 特殊情况 有趣的 取石子 注意
小约翰经常和他的哥哥玩一个非常有趣的游戏:桌子上有n堆石子,小约翰和他的哥哥轮流取石子,每个人取
的时候,可以随意选择一堆石子,在这堆石子中取走任意多的石子,但不能一粒石子也不取,我们规定取到最后一
粒石子的人算输。小约翰相当固执,他坚持认为先取的人有很大的优势,所以他总是先取石子,而他的哥哥就聪明
多了,他从来没有在游戏中犯过错误。小约翰一怒之前请你来做他的参谋。自然,你应该先写一个程序,预测一下
谁将获得游戏的胜利。
本题的输入由多组数据组成第一行包括一个整数T,表示输入总共有T组数据(T≤500)。每组数据的第一行包
括一个整数N(N≤50),表示共有N堆石子,接下来有N个不超过5000的整数,分别表示每堆石子的数目。
每组数据的输出占一行,每行输出一个单词。如果约翰能赢得比赛,则输出“John”,否则输出“Brother”
,请注意单词的大小写。
2
3
3 5 1
1
1
John
Brother
网上题解写的特别好。。。只是我看不懂而已。。。
自己凭借多年开车经验瞎写的。。。
//全为 1 的时候奇数个 1 先手必败。
//只有一个大于 1 的时候先手必胜
//也就是说现在要证明有两个及以上大于 1 的时候如果 NIM和 == 0 后手必胜。
//除去特殊情况外所有情况一定能转化到刚好有两个大于 1
//因为如果只有两堆且他们一样多的时候,显然后手必胜
//也就是说如果说两堆一样多的话就看 1 的个数是奇数还是偶数,如果是奇数则先手胜。
//如果只有两堆但是不一样多的话先手必胜。
//同理可得两堆不一样多就看 1 的个数,偶数则先手胜。
//那么最后的问题就变成了我们要判断有大于两个大于 1 的数的时候转化到上面的情况谁先谁后。
//显然 NIM和 == 0 的时候后手胜利(想一想就知道了)
/*
综上,先手必胜的条件是:
1.如果全为 1 则有偶数个 1 (NIM和 == 0)
2.只有一个大于 1
3.除上面两种情况以后 NIM和 != 0
*/
#include<bits/stdc++.h>
using namespace std;
int n, x, lin, all;
int main()
{
int T; scanf("%d", &T);
while(T--){
scanf("%d", &n);
lin = all = 0;
for(int i = 1; i <= n; ++i){scanf("%d", &x); lin += (x > 1); all ^= x;}
if(!lin && !all){printf("John\n"); continue;}
if(lin == 1){printf("John\n"); continue;}
if(lin > 1 && all){printf("John\n"); continue;}
printf("Brother\n");
}
return 0;
}
标签:sample 比赛 优势 -- 组成 特殊情况 有趣的 取石子 注意
原文地址:https://www.cnblogs.com/LLppdd/p/9892847.html