有向图游戏 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;
}
相关推荐
wengqidaifeng42 分钟前
C++从菜鸟到强手:1.基础入门
开发语言·c++
handler019 小时前
从源码到二进制:深度拆解 Linux 下 C 程序的编译与链接全流程
linux·c语言·开发语言·c++·笔记·学习
电子云与长程纠缠9 小时前
UE5 两种方式解决Decal Actor贴花拉伸问题
学习·ue5·游戏引擎
red_redemption9 小时前
自由学习记录(172)
学习·cache line 64b·重用距离
不爱吃炸鸡柳9 小时前
数据结构精讲:树 → 二叉树 → 堆 从入门到实战
开发语言·数据结构
t***5449 小时前
如何在Dev-C++中使用Clang编译器
开发语言·c++
阿荻在肝了10 小时前
Agent学习六:LangGraph学习-持久化与记忆一
python·学习·agent
Qbw200410 小时前
【Linux】进程地址空间
linux·c++
Aurorar0rua10 小时前
CS50 x 2024 Notes C - 05
java·c语言·数据结构