SG函数

首先定义mex(minimal excludant)运算,这是施加于一个集合的运算,表示最小的不属于这个集合的非负整数。例如

mex{0,1,2,4}=3、mex{2,3,5}=0、mex{}=0

对于一个给定的有向无环图,定义关于图的每个顶点的Sprague-Grundy函数g如下:g(x)=mex{ g(y) | y是x的后继},这里的g(x)即sg[x]。

例如:取石子问题,有1堆n个的石子,每次只能取{1,3,4}个石子,先取完石子者胜利,那么各个数的SG值为多少?

sg[0]=0,f[]={1,3,4},

x=1时,可以取走1-f{1}个石子,剩余{0}个,mex{sg[0]}={0},故sg[1]=1;

x=2时,可以取走2-f{1}个石子,剩余{1}个,mex{sg[1]}={1},故sg[2]=0;

x=3时,可以取走3-f{1,3}个石子,剩余{2,0}个,mex{sg[2],sg[0]}={0,0},故sg[3]=1;

x=4时,可以取走4-f{1,3,4}个石子,剩余{3,1,0}个,mex{sg[3],sg[1],sg[0]}={1,1,0},故sg[4]=2;

x=5时,可以取走5-f{1,3,4}个石子,剩余{4,2,1}个,mex{sg[4],sg[2],sg[1]}={2,0,1},故sg[5]=3; 以此类推.....

x         0  1  2  3  4  5  6  7  8....
sg[x]      0  1  0  1  2  3  2  0  1....

参考链接:SG函数和SG定理【详解】

打表:

/*
f[]:可以取走的石子个数
sg[]:0~n的sg的值
vis[]:用来求解mex{}
ps:f数组的下标,从1开始
也可以不使用ps,但是一定要保证循环有终止条件,见下面例题代码
*/
int f[N], sg[N], vis[N], ps;
void get_sg(int n)
{
    mem(sg, 0);
    for (int i = 1; i <= n; i++)
    {
        mem(vis, 0);
        for (int j = 1; f[j] <= i && j <= ps; j++)
            vis[sg[i - f[j]]] = 1;
        for (int j = 0;; j++)
            if (!vis[j])
            {
                sg[i] = j;
                break;
            }
    }
}

DFS:

/*
f[]:可以取走的石子个数,必须要升序
vis[]:用来求解mex{}
sg[]:存储sg的值
n:f[]下标
*/
const int N = 1e3 + 10;
int f[N], sg[N], n;
int sg_dfs(int x)
{
    if (~sg[x])
        return sg[x];
    int vis[N]; //必须定义在里面
    mem(vis, 0);
    for (int i = 1; i <= n; i++)
        if (x >= f[i])
        {
            sg_dfs(x - f[i]);
            vis[sg[x - f[i]]] = 1;
        }
    for (int i = 0;; i++)
        if (!vis[i])
            return sg[x] = i;
}

例题:HDU1847-Good Luck in CET-4 Everybody!

题意是只能取2的次幂,所以把2的次幂打个表,然后存进f数组,计算出每个数的sg值,然后判断是否为0

//打表
/*
f[]:可以取走的石子个数
sg[]:0~n的sg的值
vis[]:用来求解mex{}
ps:f数组的下标,从1开始
*/
const int N = 1e3 + 10;
int f[N], sg[N], vis[N];
void get_sg(int n)
{
    mem(sg, 0);
    for (int i = 1; i <= n; i++)
    {
        mem(vis, 0);
        for (int j = 1; f[j] <= i; j++)
            vis[sg[i - f[j]]] = 1;
        for (int j = 0;; j++)
            if (!vis[j])
            {
                sg[i] = j;
                break;
            }
    }
}
void init()
{
    int ans = 1, len = 0;
    for (int i = 1; i <= 11; i++)
    {
        f[++len] = ans;
        ans *= 2;
    }
    get_sg(1000);
}
int main()
{
    init();
    int x;
    while (~scanf("%d", &x))
    {
        if (sg[x])
            printf("Kiki\n");
        else
            printf("Cici\n");
    }
    return 0;
}
const int N = 1e3 + 10;
int f[N], sg[N], n;
int sg_dfs(int x)
{
    if (~sg[x])
        return sg[x];
    int vis[N]; //必须定义在里面
    mem(vis, 0);
    for (int i = 1; i <= n; i++)
        if (x >= f[i])
        {
            sg_dfs(x - f[i]);
            vis[sg[x - f[i]]] = 1;
        }
    for (int i = 0;; i++)
        if (!vis[i])
            return sg[x] = i;
}
void init()
{
    mem(sg, -1);
    sg[0] = 0;
    n = 0;
    for (int i = 1; i <= 1000; i *= 2)
        f[++n] = i;
    sg_dfs(1000);
}
int main()
{
    init();
    int x;
    while (~scanf("%d", &x))
    {
        if (sg[x])
            printf("Kiki\n");
        else
            printf("Cici\n");
    }
    return 0;
}

results matching ""

    No results matching ""