AcWing 891. Nim游戏

AcWing 891. Nim游戏

一、题目描述

给定n堆石子,两位玩家轮流操作,每次操作可以从任意一堆石子中拿走任意数量的石子(可以拿完,但不能不拿),最后无法进行操作的人视为失败。

问如果两人都采用最优策略,先手是否必胜。

输入格式
第一行包含整数 n

第二行包含 n 个数字,其中第 i 个数字表示第 i 堆石子的数量。

输出格式
如果先手方必胜,则输出 Yes

否则,输出 No

数据范围
1n105,1109

输入样例

2
2 3

输出样例

Yes

二、历史与传说

据说,它源自中国,经由被贩卖到美洲的奴工们外传。辛苦的工人们,在工作闲暇之余,用石头玩游戏以排遣寂寞。后来流传到高级人士,则用便士(Pennies),在酒吧柜台上玩。

最有名的玩法,是把十二枚便士放成345三列,拿光铜板的人赢。

后来,大家发现,先取的人只要在3那列里取走2枚,变成了145,就能稳操胜券了,游戏也就变得无趣了。

于是大家就增加列数,增加铜板的数量,这样就让人们有了毫无规律的感觉,不易于把握。

三、博弈论结论

在两名选手都足够聪明,每一步都是最优解的情况下:

a1 ^ a2 ^ ... ^an=0 先手必败
a1 ^ a2 ^ ... ^an0 先手必胜
就是把所有堆的石子个数异或起来,结果是零,先手必败,不是零,先手必胜。

四、证明过程

1、结论1

操作到最后时,每堆石子数都是0,有0 ^ 0 ^ 0=0

2、结论2

在操作过程中,如果 a1a2an=x0。那么玩家必然可以通过拿走某一堆若干个石子将异或结果变为0

证明

  • 不妨设结果x的二进制表示中最高一位1在第k
  • 那么在a1,a2,,an中,必然有一个数ai,它的第k位是1(要不那个1从哪里来的?),且aix<ai:
  • 让第i堆石子保留aix个:即从第i堆石子中拿走ai(aix)个石子
  • 此时新状态的异或和:

    a1a2aian=a1a2(aix)an=a1a2ai...anx=xx=0

证毕

解释:也就是说,不管给我啥样的情况,我都能找出一种办法,使得一次拿取后满足异或和为0

其实,我们也并不非得要找到某个有明显特征(指x的最高位1在第k位,而ai的第k位是1)的ai,更通用的作法是

for (int i = 0; i < n; i++) {
    if((a[i] ^ x) < a[i]){
      1. 拿走 a[i]-(a[i] ^ x)个
      2. 剩下 a[i] ^ x个
    }
}

这样的话,我们也就可以求出有多少种拿法,并且知道每种拿法是拿走多少,剩下多少。

3、结论3

在操作过程中,如果 a1a2an=0,那么无论玩家怎么拿,必然会导致最终异或结果不为0

反证法:假设玩家从第i堆石子拿走若干个,结果仍是0。不妨设还剩下ai个,因为不能1个都不拿,所以0ai<ai,且a1a2aian=0

进行构造

(a1a2aian)(a1a2aian)

整体构造完式子后,利用异或运算的 结合律 性质,让

a1a1=0,a2a2=0,a3a3=0,...

aia=0

ai=a,与假设0a<ai矛盾。

证毕

基于上述三个结论

  1. 如果先手面对的局面是x=a1a2an0,那么先手总可以通过拿走某一堆若干个石子(aix个),将局面变成a1a2an=0。如此重复,最后一定是后手面临最终没有石子可拿的状态。先手必胜。

  2. 如果先手面对的局面是a1a2an=0,那么无论先手怎么拿,都会将局面变成a1a2an0,那么后手总可以通过拿走某一堆若干个石子,将局面变成a1a2an=0。如此重复,最后一定是先手面临最终没有石子可拿的状态,先手必败。

五、实现代码

#include <bits/stdc++.h>

using namespace std;

int main() {
    int n;
    cin >> n;

    int res = 0; // 起始值是0,因为任何数与0进行异或都它本身
    while (n--) {
        int x;
        cin >> x;
        res ^= x;
    }
    if (res)
        puts("Yes"); // 异或值非零,必胜
    else
        puts("No"); // 异或值是零,必败
    return 0;
}
posted @   糖豆爸爸  阅读(336)  评论(0编辑  收藏  举报
编辑推荐:
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
阅读排行:
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 【自荐】一款简洁、开源的在线白板工具 Drawnix
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· Docker 太简单,K8s 太复杂?w7panel 让容器管理更轻松!
历史上的今天:
2019-10-11 IDEA中css文件包红色下划线
2019-10-11 关于Idea中右边的maven projects窗口找不到了如何调出来
2019-10-11 如何解决IntelliJ idea的maven工程提示的Cannot Resolve Symbol
2018-10-11 LINUX中目录大小查看
Live2D
点击右上角即可分享
微信分享提示