C++ 数论相关题目,博弈论,SG函数,集合-Nim游戏

给定 n

堆石子以及一个由 k

个不同正整数构成的数字集合 S

现在有两位玩家轮流操作,每次操作可以从任意一堆石子中拿取石子,每次拿取的石子数量必须包含于集合 S

,最后无法进行操作的人视为失败。

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

输入格式

第一行包含整数 k

,表示数字集合 S

中数字的个数。

第二行包含 k

个整数,其中第 i

个整数表示数字集合 S

中的第 i

个数 si

第三行包含整数 n

第四行包含 n

个整数,其中第 i

个整数表示第 i

堆石子的数量 hi

输出格式

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

否则,输出 No。

数据范围

1≤n,k≤100

,

1≤si,hi≤10000

输入样例:

2

2 5

3

2 4 7

输出样例:

Yes

SG函数:表示当前状态所不能到达状态中最小的自然数。

必胜状态:SG不等于0;

必败状态:SG等于0。

如果有多个图,将每个初始的SG值异或,等于0必败,不等于0必胜。

cpp 复制代码
#include <iostream>
#include <cstring>
#include <algorithm>
#include <unordered_set>

using namespace std;

const int M = 110, N = 10010;

int m, n;
int s[M], f[N]; //s存可以取的数,f表明一个状态的sg值,一个状态是一个数,一个确定石子个数的堆可以分解成一个图表示状态。

int sg(int x)
{
    if(f[x] != -1) return f[x]; //避免重复计算,如果x状态算过的话,就直接返回这个状态的sg值
    
    unordered_set<int> S;//存能到达的状态的sg值。
    for(int i = 0; i < m; i ++ ) //遍历每一个图(堆,石子堆)
        if(x >= s[i])
            S.insert(sg(x - s[i]));
    
    for(int i = 0; ; i ++ )
        if(!S.count(i)) //找到最小的不存在的状态自然数,说明当前状态的sg值就是i这个数
            return f[x] = i;
    
}

int main ()
{
    cin>>m;
    for(int i = 0; i < m; i ++ ) cin>>s[i];
    
    cin>>n;
    
    memset(f, -1, sizeof f);
    
    int res = 0;
    while(n -- )
    {
        int x;
        cin>>x;
        res ^= sg(x);
    }
    
    if(res) puts("Yes");
    else puts("No");
    
    return 0;
}
相关推荐
星释几秒前
Rust 练习册 :Nucleotide Codons与生物信息学
开发语言·算法·rust
熊猫钓鱼>_>5 分钟前
Java面向对象核心面试技术考点深度解析
java·开发语言·面试·面向对象··class·oop
她说彩礼65万12 分钟前
C# 代理模式
开发语言·c#·代理模式
点金石游戏出海16 分钟前
每周资讯 | 印度数字媒体与娱乐市场在2025财年达93亿美;《崩坏:星穹铁道》新版本登顶iOS畅销榜首
游戏·娱乐·媒体·游戏资讯·崩坏星穹铁道
程序员大雄学编程20 分钟前
用Python来学微积分34-定积分的基本性质及其应用
开发语言·python·数学·微积分
liu****25 分钟前
12.线程(二)
linux·开发语言·c++·1024程序员节
DKPT36 分钟前
如何设置JVM参数避开直接内存溢出的坑?
java·开发语言·jvm·笔记·学习
林一百二十八1 小时前
Python实现手写数字识别
开发语言·python
小小鱼儿飞1 小时前
QT Quick QML项目音乐播放器16----无边框窗口拖动、小窗播放、隐藏系统托盘
开发语言·qt
-指短琴长-1 小时前
Qt的下载和安装【Windows】
开发语言·windows·qt