一 . nim游戏的数学定义
Nim游戏是博弈论中最经典的模型,它又有着十分简单的规则和无比优美的结论 。
Nim游戏是组合游戏(Combinatorial Games)的一种,准确来说,属于“Impartial Combinatorial Games”(以下简称ICG)。
一般而言,满足以下条件的游戏是ICG:
对于条件3,有更进一步的定义Position。
我们将Position分为两类:
P-position:在当前的局面下,先手必败。
N-position:在当前的局面下,先手必胜。
他们有如下性质:
1.合法操作集合为空的局面是P-position;
2.可以移动到P-position的局面是N-position;
3.所有移动都只能到N-position的局面是P-position。
重要结论:对于一个Nim游戏的局面
,它是P-position时,当且仅当
^
^...^
=0;它是N-position时,当且仅当
^
^...^
!=0,其中^表示异或(xor)运算。
甲,乙两个人玩 nim 取石子游戏。
nim 游戏的规则是这样的:地上有 n 堆石子(每堆石子数量小于
),每人每次可从任意一堆石子里取出任意多枚石子扔掉,可以取完,不能不取。每次只能从一堆里取。最后没石子可取的人就输了。假如甲是先手,且告诉你这 n 堆石子的数量,他想知道是否存在先手必胜的策略。
本题有多组测试数据。
第一行一个整数 T (
),表示有 T 组数据
接下来每两行是一组数据,第一行一个整数 n,表示有 n 堆石子,
。
第二行有 n 个数,表示每一堆石子的数量.
共 T 行,每行表示如果对于这组数据存在先手必胜策略则输出 Yes,否则输出 No。
输入 #1
2 2 1 1 2 1 0
输出 #1
No Yes
- #include
- #include
- #include
- #include
- #include
- using namespace std;
- inline int read()
- {
- int x=0,f=1;
- char c=getchar();
- while(c<'0'||c>'9')
- {
- if(c=='-') f=-1;
- c=getchar();
- }
- while(c>='0' && c<='9')
- {
- x=(x<<3)+(x<<1)+(c^48);
- c=getchar();
- }
- return x*f;
- }
-
- int main()
- {
- int t=read();
- for(int i=1;i<=t;++i)
- {
- int n=read(),res;
- for(int j=1;j<=n;++j)
- {
- int a=read();
- if(j == 1) res=a; //将第一个数记录在res中
- else res^=a; //后添加的数与前面的数不断异或
- }
- if(res == 0) printf("No\n");
- else printf("Yes\n");
- }
- }