博弈论-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();
    }



}
相关推荐
xian_wwq18 小时前
【学习笔记】大模型微调实战指南
笔记·学习·微调·llama factory
张人玉18 小时前
PyCharm配置PyQt5安装报错详情笔记
笔记·qt·pycharm·pyqt5
得闲喝茶18 小时前
JavaScript在数据处理的应用
开发语言·前端·javascript·经验分享·笔记
O&REO18 小时前
哈尔滨工业大学 837 网安自命题开源资料+笔记+经验贴
笔记
晓梦林18 小时前
Qingmei靶场学习笔记
笔记·学习·安全·web安全
kdxiaojie18 小时前
U-Boot分析【学习笔记】(6)
linux·笔记·学习
armwind18 小时前
这6年的小感悟-重新记录自己
笔记
晓梦林18 小时前
Commit靶场学习笔记
笔记·学习·安全·web安全
sheeta199819 小时前
LeetCode 每日一题笔记 日期:2026.05.10 题目:2770. 达到末尾下标所需的最大跳跃次数
笔记·算法·leetcode
LeeeX!19 小时前
OpenClaw CLI 完整实操笔记
笔记·openclaw