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



}
相关推荐
handler0113 分钟前
Linux: 基本指令知识点(2)
linux·服务器·c语言·c++·笔记·学习
南境十里·墨染春水4 小时前
C++ 笔记 thread
java·开发语言·c++·笔记·学习
南境十里·墨染春水4 小时前
C++ 笔记 高级线程同步原语与线程池实现
java·开发语言·c++·笔记·学习
lkforce4 小时前
MiniMind学习笔记(二)--model_minimind.py
笔记·python·学习·minimind·minimindconfig
三品吉他手会点灯5 小时前
C语言学习笔记 - 1.C概述 - 本讲内容概述
c语言·笔记·学习
是孑然呀7 小时前
【笔记】激光定位-激光切割指针偏移设置
笔记
中屹指纹浏览器8 小时前
2026指纹浏览器性能优化实战:多开稳定性与资源占用控制全解析
经验分享·笔记
一只大袋鼠9 小时前
MyBatis 特性(三):缓存、延迟加载、注解开发
java·数据库·笔记·sql·缓存·mybatis
三品吉他手会点灯9 小时前
C语言学习笔记 - 3. C概述 - C语言的起源和发展
c语言·笔记·学习
爱莉希雅&&&9 小时前
MySQL MGR 组复制 完整笔记
linux·数据库·笔记·mysql·mgr·数据库同步