acwing算法基础之数学知识--Nim游戏和集合Nim游戏

目录

  • [1 基础知识](#1 基础知识)
  • [2 模板](#2 模板)
  • [3 工程化](#3 工程化)

1 基础知识

(一)

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

结论:如果这n个数异或之和为0,则先手必败,否则先手必胜。

代码表示为,

cpp 复制代码
#include <iostream>

using namespace std;

int main() {
    int n;
    cin >> n;
    int res = 0;
    while (n--) {
        int x;
        cin >> x;
        res = res ^ x;
    }
    
    if (res) puts("Yes");
    else puts("No");
    
    return 0;
}

(二)

集合Nim游戏:在Nim游戏的基础上,对每次取走的石子做了限制,每次取走的石子数必须在集合 S S S内。判断是否先手必胜。

抽象建模为,

有向图游戏和SG函数:在一个有向无环图中,只有一个起点,上面有一个棋子,两个玩家轮流沿有向边推动棋子,不能走的玩家判负。

定义mex函数的值为,不属于集合S中的最小非负整数,即:
m e x ( S ) = m i n { x } ( x ∉ S , x ∈ N ) mex(S)=min\{x\} \ (x\notin S, x\in N) mex(S)=min{x} (x∈/S,x∈N)

例如mex({0,2,3}) = 1, mex({1,2}) = 0。

对于状态 x x x和它的所有 k k k个后继状态 y 1 , y 2 , ⋯   , y k y_1,y_2,\cdots,y_k y1,y2,⋯,yk,定义SG函数:
S G ( x ) = m e x { S G ( y 1 ) , S G ( y 2 ) , ⋯   , S G ( y k ) } SG(x)=mex\{SG(y_1), SG(y_2), \cdots, SG(y_k)\} SG(x)=mex{SG(y1),SG(y2),⋯,SG(yk)}

而对于由n个有向图组成的组合游戏,设它们的起点分别为 s 1 , s 2 , ⋯   , s n s_1,s_2,\cdots,s_n s1,s2,⋯,sn,则有定理:当且仅当这 n n n个数 S G ( s 1 ) , S G ( s 2 ) , ⋯   , S G ( s n ) SG(s_1),SG(s_2),\cdots,SG(s_n) SG(s1),SG(s2),⋯,SG(sn)的异或和不为0时,这个游戏是先手必胜的,否则,是先手必败的。

C++代码如下,

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

using namespace std;

const int N = 110, M = 1e4 +10;
int n, m;
int s[N]; //每次可以取的石子数目
int f[M]; //这堆有x个石子,求sg[x]的值

int sg(int x) {
    if (f[x] != -1) return f[x];
    
    unordered_set<int> S;//x能走到的结点的sg函数值
    for (int i = 0; i < n; ++i) {
        if (x - s[i] >= 0) S.insert(sg(x-s[i]));
    }
    
    for (int i = 0; ; ++i) {
        if (S.count(i) == 0) {
            f[x] = i;
            break;
        }
    }
    return f[x];
}

int main() {
    cin >> n;
    for (int i = 0; i < n; ++i) cin >> s[i];
    
    int res = 0;
    memset(f, -1, sizeof f);
    cin >> m;
    while (m--) {
        int x;
        cin >> x;
        res ^= sg(x);
    }
    
    if (res) puts("Yes");
    else puts("No");
    
    return 0;
}

2 模板

暂无。。。

3 工程化

题目1:拆分Nim游戏,取走一堆,放回两堆规模更小的石子。

解题思路:重点在于如何确认某一堆的sg值,这样考虑遍历两堆规模更小的石子,就是它的下一步状态,求得它们的sg值,进行mex操作,即可得到这堆石子的sg值。

C++代码如下,

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

using namespace std;

const int N = 110;

int n;
int f[N]; //sg值

int sg(int x) {
    if (f[x] != -1) return f[x];
    
    //x可以走到的状态的sg值
    unordered_set<int> S;
    for (int i = 0; i < x; ++i) {
        for (int j = 0; j <= i; ++j) {
            S.insert(sg(i) ^ sg(j));
        }
    }
    
    //mex操作
    for (int i = 0; ; ++i) {
        if (!S.count(i)) {
            return f[x] = i;
        }
    }
}

int main() {
    memset(f, -1, sizeof f);
    
    cin >> n;
    int res = 0;
    for (int i = 0; i < n; ++i) {
        int x;
        cin >> x;
        res ^= sg(x);
    }
    
    if (res) puts("Yes");
    else puts("No");
    
    return 0;
}
相关推荐
爱coding的橙子7 分钟前
每日算法刷题Day70:10.13:leetcode 二叉树10道题,用时2h
算法·leetcode·深度优先
ghie909033 分钟前
基于MATLAB的遗传算法优化支持向量机实现
算法·支持向量机·matlab
wanhengidc1 小时前
巨椰云手机引领未来
运维·服务器·网络·游戏·智能手机
wanhengidc1 小时前
云手机的真实体验感怎么样
运维·服务器·安全·游戏·智能手机
朝新_1 小时前
【优选算法】第一弹——双指针(上)
算法
艾莉丝努力练剑2 小时前
【C++STL :stack && queue (一) 】STL:stack与queue全解析|深入使用(附高频算法题详解)
linux·开发语言·数据结构·c++·算法
CoovallyAIHub2 小时前
ICLR 2026 惊现 SAM 3,匿名提交,实现“概念分割”,CV领域再迎颠覆性突破?
深度学习·算法·计算机视觉
wanhengidc2 小时前
云手机 流畅运行
运维·服务器·安全·游戏·智能手机
IT古董2 小时前
【第五章:计算机视觉-计算机视觉在工业制造领域中的应用】1.工业缺陷分割-(2)BiseNet系列算法详解
算法·计算机视觉·制造
电鱼智能的电小鱼2 小时前
服装制造企业痛点解决方案:EFISH-SBC-RK3588 预测性维护方案
网络·人工智能·嵌入式硬件·算法·制造