有向图游戏 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;
}
相关推荐
诚丞成44 分钟前
计算世界之安生:C++继承的文水和智慧(上)
开发语言·c++
清梦20201 小时前
经典问题---跳跃游戏II(贪心算法)
算法·游戏·贪心算法
Dream_Snowar1 小时前
速通Python 第四节——函数
开发语言·python·算法
1nullptr1 小时前
三次翻转实现数组元素的旋转
数据结构
Altair澳汰尔1 小时前
数据分析和AI丨知识图谱,AI革命中数据集成和模型构建的关键推动者
人工智能·算法·机器学习·数据分析·知识图谱
OopspoO1 小时前
qcow2镜像大小压缩
学习·性能优化
TT哇1 小时前
【数据结构练习题】链表与LinkedList
java·数据结构·链表
东风吹柳2 小时前
观察者模式(sigslot in C++)
c++·观察者模式·信号槽·sigslot
A懿轩A2 小时前
C/C++ 数据结构与算法【栈和队列】 栈+队列详细解析【日常学习,考研必备】带图+详细代码
c语言·数据结构·c++·学习·考研·算法·栈和队列
Python机器学习AI2 小时前
分类模型的预测概率解读:3D概率分布可视化的直观呈现
算法·机器学习·分类