有向图游戏 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;
}
相关推荐
lichuangcsdn5 分钟前
【springcloud学习(dalston.sr1)】项目整体介绍(含源代码)(一)
学习·spring·spring cloud
红衣小蛇妖10 分钟前
Python基础学习-Day23
开发语言·python·学习
June`18 分钟前
专题四:综合练习( 找出所有子集的异或总和再求和)
c++·算法·深度优先·剪枝
越甲八千22 分钟前
windowsC++操作ADB
c++·windows·adb
大白的编程日记.22 分钟前
【Linux学习笔记】理解一切皆文件实现原理和文件缓冲区
linux·笔记·学习
孞㐑¥23 分钟前
Linux之进程控制
linux·开发语言·c++·经验分享·笔记
小石(努力版)26 分钟前
嵌入式STM32学习——外部中断EXTI与NVIC的基础练习⭐
stm32·单片机·学习
Magnum Lehar30 分钟前
3d游戏引擎的Utilities模块实现下
c++·算法·游戏引擎
莹莹学编程—成长记32 分钟前
list基础用法
数据结构·list
一丝晨光35 分钟前
数值溢出保护?数值溢出应该是多少?Swift如何让整数计算溢出不抛出异常?类型最大值和最小值?
java·javascript·c++·rust·go·c·swift