博弈论-Nim游戏

题目描述

蓝桥公司给他们的员工准备了丰厚的奖金,公司主管小明并不希望发太多的奖金,他想把奖金留给智慧的人,于是他决定跟每一个员工玩一个游戏,规则如下:

  • 桌面上一共有 n 堆一元钱。
  • 双方轮流行动,由小明先行动,每次行动从某一堆钱中拿走若干元(至少一元钱),取走最后一元钱的人获胜。

请问员工们能拿到奖金吗?

输入描述

第一行为一个整数 T,表示测试数据数量。

每个测试用例包含俩行。第一行为一个整数 n , 第二行包括 n 个整数 a1,a2...an​ 表示第 i 堆有 ai​ 元。

保证所有测试用例的 nn 的和不超过 2×1052×105。

输出描述

如果员工能拿到奖金输出 YES​ , 否则输出 NO

输入输出样例

示例

输入

复制代码
3
2
1 1
1
1
3
2 2 1

输出

复制代码
YES
NO
NO

Nim游戏:

简介

有n堆不同的物品,这些堆中各自物品的数量是任意的。两名玩家轮流取走堆中物品,可以选择将某一堆中任意数量的物品拿走,至少1个,至多全部拿走,但不能不拿或跨堆拿取物品。最终,拿到最后一个物品的玩家获胜。

结论

若初态为必胜态(a1^a2^a3^......^an != 0),则先手必胜;

若初态为必败态(a1^a2^a3^......^an == 0),则先手必败。

两个定理

  1. 必胜态的后继状态至少存在一个必败态;
  2. 必败态的后继状态均为必胜态。

总结

如果先手获得的是必胜态(所有堆数量异或结果!=0),根据定理1,先手可以在一个堆中取一定数量使得给到对手的初态是必败态,又根据定理2,对手不论怎么取,最终回到你手上时仍然为必胜态(如果我初状态为必胜态,则我一定可以给下一个人制造一个必败态)。

代码实现

java 复制代码
import java.util.*;


public class Main {

    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);

        int T=sc.nextInt();
        for (int i = 0; i < T; i++) {
            int n=sc.nextInt();
            int num=sc.nextInt();
            for (int j = 1; j < n; j++) {
                num^=sc.nextInt();
            }

            if (num==0) System.out.println("YES");
            else System.out.println("NO");
        }

        sc.close();
    }



}
相关推荐
Hello_Embed2 小时前
嵌入式上位机开发入门(五):UDP 编程 —— Server 端实现
笔记·单片机·网络协议·udp·嵌入式
热水过敏2 小时前
前路迷茫,再次起航
笔记·程序人生·职场和发展
wanderist.2 小时前
高维矩阵的压维存储和高维差分
c++·算法·蓝桥杯
迈巴赫车主2 小时前
蓝桥杯192.等差数列java
java·数据结构·算法·职场和发展·蓝桥杯
chase。2 小时前
【学习笔记】RoboForge:让文本指令“落地”到人形机器人——一个物理优化与隐式驱动的端到端框架
笔记·学习·机器人
chase。2 小时前
【学习笔记】从经典算法到通用神经运动规划器
笔记·学习·算法
牛奶咖啡133 小时前
免费笔记软件且优先本地私有化——Joplin、Obsidian
笔记·obsidian·joplin·待办事项应用程序·开源笔记应用·可私有化本地笔记应用·笔记的同步与插件安装
忙什么果3 小时前
transformer学习笔记1
笔记·学习·transformer
阿i索3 小时前
【蓝桥杯备赛Day5】排序
笔记·蓝桥杯·排序算法