算法学习笔记(Nim游戏)

server/2024/10/19 13:20:58/

N i m Nim Nim游戏

n n n堆物品,每堆有 a i a_i ai个,每个玩家轮流取走任意一堆的任意个物品,但不能不取,取走最后一个物品的人获胜。

N i m Nim Nim游戏是一种经典的公平组合游戏。现在对它进行分析。

首先定义两个博弈中的状态:

  • 必胜状态:先手必胜的状态。
  • 必败状态:先手必败的状态。

对于这两个状态,我们可以知道:

  1. 没有后继状态的状态必然是必败状态。在这个状态中先手的是败者,因为他无法通过操作将游戏进行下去了。
  2. 一个状态是必胜状态当且仅当存在至少一个必败状态为它的后继状态。在这个状态中先手的人可以通过一次操作让对手在必败状态中先手。
  3. 一个状态的所有后继状态均为必胜状态,那么这个状态为必败状态。在这个状态中先手,无法避免让对方在必胜状态中先手。

回到 N i m Nim Nim游戏:

N i m Nim Nim游戏中,一个很显然的必败状态就是所有物品堆中物品的数量都为 0 0 0,即 [ 0 , 0 , . . . , 0 ] [0, 0, ..., 0] [0,0,...,0]。这个状态也是最终态。可以知道,在最终态时,所有物品堆中的物品数量的异或和是等于 0 0 0的,我们不妨假设状态和物品数量的异或和有关系。

证明有关:

一个非 0 0 0的异或和,产生最高位的 1 1 1总需要有奇数个数字来提供对应位置的 1 1 1。而我们为了消去这个 1 1 1,可以选择任意一个提供这个 1 1 1的数字,使其二进制中该位上的数字为 0 0 0,而且修改最高位为 0 0 0后得到的数字永远小于原来的数字,也就是说,我们可以任意修改其他位上的数字从而使得全部物品数量的异或和为 0 0 0

而对于一个为 0 0 0的异或和,假设存在一个 b ≠ a i b \not = a_i b=ai使得我们将 a i a_i ai修改为 b b b后,异或和还是为 0 0 0,则有 0 ⊕ a i ⊕ b = 0 0 \oplus a_i \oplus b = 0 0aib=0,为了使这个式子成立 b b b就要等于 a i a_i ai,与假设违背。

换句话说,对于一个物品数量异或和不为 0 0 0的状态,我们可以通过一次操作将物品数量的异或和修改为 0 0 0,而对于一个物品数量异或和为 0 0 0的操作,我们无法只通过一次操作保持物品数量的异或和不变。

从上可以得出,在 N i m Nim Nim游戏中,物品数量异或和为 0 0 0的状态是必败状态,物品数量异或和不为 0 0 0的状态是必胜状态。

接下来看例题:

【模板】Nim 游戏

【模板】Nim 游戏

题目描述

甲,乙两个人玩 nim 取石子游戏。

nim 游戏的规则是这样的:地上有 n n n 堆石子(每堆石子数量小于 1 0 4 10^4 104),每人每次可从任意一堆石子里取出任意多枚石子扔掉,可以取完,不能不取。每次只能从一堆里取。最后没石子可取的人就输了。假如甲是先手,且告诉你这 n n n 堆石子的数量,他想知道是否存在先手必胜的策略。

输入格式

本题有多组测试数据。

第一行一个整数 T T T T ≤ 10 T\le10 T10),表示有 T T T 组数据

接下来每两行是一组数据,第一行一个整数 n n n,表示有 n n n 堆石子, n ≤ 1 0 4 n\le10^4 n104

第二行有 n n n 个数,表示每一堆石子的数量.

输出格式

T T T 行,每行表示如果对于这组数据存在先手必胜策略则输出 Yes,否则输出 No

样例 #1

样例输入 #1

2
2
1 1
2
1 0

样例输出 #1

No
Yes

根据刚才的推论,我们只需要计算所有数字的异或和,就可以得出先手时处在必胜状态还是必败状态。用 O ( n ) O(n) O(n)的复杂度即可得出最后的胜负结果。

#include<bits/stdc++.h>
using namespace std;void solve()
{int n; cin >> n;int ans = 0;for(int i = 1; i <= n; ++i){int x; cin >> x;ans ^= x;}cout << (ans ? "Yes" : "No") << '\n';
}int main()
{ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);int _; cin >> _;while(_--) solve();return 0;
}

http://www.ppmy.cn/server/41411.html

相关文章

qt的http原理

#ifndef TURING_H #define TURING_H #include <QObject》 #include <QNetworkAccessManager》 #include <QNetworkRequest》 #include <QNetworkReply》 class Turing : public QObject { Q_OBJECT public: explicit Turing(QObject *parent 0);Q_INVOKABLE v…

LeetCode力扣第114题:多种算法实现 将二叉树展开为链表

作者介绍&#xff1a;10年大厂数据\经营分析经验&#xff0c;现任大厂数据部门负责人。 会一些的技术&#xff1a;数据分析、算法、SQL、大数据相关、python 欢迎加入社区&#xff1a;码上找工作 作者专栏每日更新&#xff1a; LeetCode解锁1000题: 打怪升级之旅 python数据分析…

安装InternVL

InternVL 官网 interVL 安装 完全删除环境和环境中的所有软件包 conda remove -n env_name --all 安装 克隆此存储库&#xff1a; git clone https://github.com/OpenGVLab/InternVL.git 创建conda虚拟环境并激活&#xff1a; conda create -n inter pytho…

竞赛课第十周(巴什游戏,尼姆博弈)

目录 目的: 实验内容: 第一题 思路&#xff1a; 【参考代码】 【运行结果】 第二题 输入&#xff1a; 输出&#xff1a; 【参考代码】 【运行结果】 目的: 熟悉并掌握公平组合游戏 &#xff08;1&#xff09;巴什游戏、尼姆游戏 &#xff08;2&#xff09;图游戏…

【Pytorch】7.使用Module模块搭建简易神经网络

什么是Moudel模块 torch.nn中的module是PyTorch中用于构建神经网络模型的基本单元。它包含了各种神经网络层、激活函数、损失函数等&#xff0c;可以通过组合不同的module来构建复杂的神经网络模型。每个module都包含了参数和方法&#xff0c;可以进行前向传播和反向传播等操作…

面试分享——Elasticsearch面试题

目录 1.Elasticsearch数据建模相关问题 1.1问题描述 1.2问题回答 2.Elasticsearch 查询和分析相关问题 2.1问题描述 2.2问题回答 3.Elasticsearch 集成与开发问题 3.1问题描述 3.2问题回答 4.Elasticsearch DSL 相关应用选型等问题 4.1问题描述 4.2问题回答 4.2.1…

【Web】CTFSHOW 七夕杯 题解

目录 web签到 easy_calc easy_cmd web签到 CTF中字符长度限制下的命令执行 rce(7字符5字符4字符)汇总_ctf中字符长度限制下的命令执行 5个字符-CSDN博客7长度限制直接梭了 也可以打临时文件RCE import requestsurl "http://4ae13f1e-8e42-4afa-a6a6-1076acd08211.c…

DataLab-数据分析的Ai辅助工具

添加图片注释&#xff0c;不超过 140 字&#xff08;可选&#xff09;DataLab是一个由DataCamp提供的强大在线数据分析平台&#xff0c;它通过AI技术简化了数据处理流程&#xff0c;使得用户无需编程或数据分析的高级技能即可快速获取数据洞察。它支持多种数据源&#xff0c;包…