有向图游戏 SG函数【博弈论】C++

SG函数可以用来判断在一个给定的有向图游戏中,当前局面的胜负状态。

SG函数的定义如下:

设当前节点为v,那么SG(v)为当前局面的SG值。SG(v)的定义如下:

  • 如果当前节点v没有后继节点,则SG(v) = 0

  • 如果当前节点v有若干个后继节点,分别为v1,v2,...,v_n,那么SG(v)为所有后继节点的SG值的异或和。即:SG(v) = SG(v1) XOR SG(v2) XOR ... XOR SG(v_n)

根据SG函数的定义,可以得出以下结论:

  • 如果SG(v)为0,则当前局面为必败态(先手必输);

  • 如果SG(v)不为0,则当前局面为必胜态(先手必胜);

  • SG函数具有性质:SG(v) = SG(w),当且仅当v和w的后继节点的SG值相同。

需要注意的是,有向图游戏中的SG函数只适用于无环图。对于有环图,SG函数的定义需要进一步扩展。

cpp 复制代码
#include <bits/stdc++.h>
#define ll long long
#define ull unsigned long long
// #define int long long
#define endl "\n"
#define KUI ios::sync_with_stdio(0), cin.tie(0), cout.tie(0)
using namespace std;
const int con = 2e5 + 4;
const int mod = 998244353;
int n, m, k, f[con];
vector<int> v[con];
int sg(int u)
{
    if (f[u] != -1)
    {
        return f[u];
    }
    set<int> s;
    for (auto x : v[u])
    {
        s.insert(sg(x));
    }
    int ans = 0;
    while (1)
    {
        if (s.count(ans) == 0)
        {
            return f[u] = ans;
        }
        ans++;
    }
}
void take()
{
    cin >> n >> m >> k;
    for (int i = 1; i <= m; i++)
    {
        int a1, a2;
        v[a1].push_back(a2);
    }
    memset(f, -1, sizeof f);
    int res = 0;
    int x;
    for (int i = 1; i <= k; i++)
    {
        cin >> x;
        res ^= sg(x);
    }
    if (res > 0)
    {
        cout << "先手必胜" << endl;
    }
    else
    {
        cout << "后手必胜" << endl;
    }
}
signed main()
{
    KUI;
    int t1 = 1;
    while (t1--)
    {
        take();
    }
    return 0;
}
相关推荐
hd51cc12 小时前
MFC消息 学习笔记
笔记·学习·mfc
Dev7z13 小时前
基于MATLAB数学形态学的边缘检测算法仿真实现
算法·计算机视觉·matlab
kyle~16 小时前
C++---value_type 解决泛型编程中的类型信息获取问题
java·开发语言·c++
盐焗西兰花18 小时前
鸿蒙学习实战之路:状态管理最佳实践
学习·华为·harmonyos
NiNi_suanfa19 小时前
【Qt】Qt 批量修改同类对象
开发语言·c++·qt
信奥胡老师19 小时前
苹果电脑(mac系统)安装vscode与配置c++环境,并可以使用万能头文件全流程
c++·ide·vscode·macos·编辑器
妖灵翎幺19 小时前
C++ 中的 :: 操作符详解(一切情况)
开发语言·c++·ide
风筝在晴天搁浅20 小时前
代码随想录 718.最长重复子数组
算法
kyle~20 小时前
算法---回溯算法
算法
小毅&Nora20 小时前
【人工智能】【深度学习】 ⑦ 从零开始AI学习路径:从Python到大模型的实战指南
人工智能·深度学习·学习