POJ 2960 S-Nim(SG函数模板题)

链接:https://vjudge.net/problem/POJ-2960

题意:每行输入首先给出一个数k,代表集合S的大小,接下来紧跟着k个数,表示集合S里的数。接下来一行数为m代表有m个游戏,后面m行每行第一个数字为n代表有n堆石子,后面紧跟着n个数代表每堆石子的个数。多组数据,做到0结束

   对于每组数据,我们要输出n个字母,第i个字母为“W”代表第i个游戏先手必胜,“L”代表第i个游戏先手必败,做完一组数据后换行。

题解:模板题

#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;

// sg值的计算方法:
//1.可选步数为1~m的连续整数,直接取模即可,SG(x) = x % (m+1);‘
//2.可选步数为任意步,SG(x) = x
//3.可选步数为一系列不连续的数,用打表或者dfs计算
const int maxn=105, maxm=10005;
int S[maxn], sg[maxm], vis[maxm];
int n, m, size;
void get_sg()
{
    memset(sg, 0, sizeof(sg));
    for(int i=1; i<=maxm; i++)
    {
        memset(vis, 0, sizeof(vis));
        for(int j=0; j<size; j++)
        {
            if(i<S[j]) break;
            vis[sg[i-S[j]]]=1;
        }
        for(int j=0; j<=maxn; j++)
            if(!vis[j]){
                sg[i]=j; break;
            }
    }
}

int dfs_sg(int x)
{
    if(sg[x]!=-1) return sg[x];
    memset(vis, 0, sizeof(vis));
    for(int i=0; i<n; i++)
    {
        if(x>=S[i])
        {
            dfs_sg(x-S[i]);
            vis[sg[x-S[i]]]=1;
        }
    }
    int res;
    for(int i=0; ;i++)
        if(!vis[i]){
            res=i; break;
        }
    return sg[x]=res;
}

int main()
{
    //freopen("in.txt", "r", stdin);
    while(scanf("%d", &size), size)
    {
        for(int i=0; i<size; i++)
            scanf("%d", &S[i]);
        sort(S, S+size);
        get_sg();
        scanf("%d", &n);
        for(int i=0; i<n; i++)
        {
            scanf("%d", &m);
            int ans=0, t;
            for(int i=0; i<m; i++)
            {
                scanf("%d", &t);
                ans^=sg[t];
            }
            printf(ans? "W":"L");
        }
        printf("\n");
    }
    return 0;
}

原文地址:https://www.cnblogs.com/Yokel062/p/11664206.html

时间: 2024-10-06 11:32:59

POJ 2960 S-Nim(SG函数模板题)的相关文章

hdu 1536 S-Nim|| poj 2960 S-Nim (sg函数)

#include <stdio.h> #include <string.h> int s[110]; int sg[10010],hash[110]; int n, m; int getsg(int x) //sg模板 { int i; if(sg[x] != -1) return sg[x]; memset(hash,0,sizeof(hash)); for(i = 0; i < n; i++) { if(x >= s[i]) { sg[x - s[i]] = get

POJ 2960 S-Nim 博弈论 sg函数

http://poj.org/problem?id=2960 sg函数几乎是模板题. 调试代码的最大障碍仍然是手残在循环里打错变量名,是时候换个hydra产的机械臂了[超想要.jpg] 1 #include<cstdio> 2 #include<cstring> 3 #include<algorithm> 4 #include<cmath> 5 #include<iostream> 6 #include<map> 7 using na

poj 2960 S-Nim(SG函数)

S-Nim Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 3694   Accepted: 1936 Description Arthur and his sister Caroll have been playing a game called Nim for some time now. Nim is played as follows: The starting position has a number of h

poj 3692 Kindergarten (最大团模板题)

题目链接:http://poj.org/problem?id=3692 Kindergarten Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 5156   Accepted: 2512 Description In a kindergarten, there are a lot of kids. All girls of the kids know each other and all boys also know e

HDU 3032 Nim or not Nim?(sg函数)

题目链接 暴力出来,竟然眼花了以为sg(i) = i啊....看表要认真啊!!! #include <cstdio> #include <cstring> #include <iostream> using namespace std; #define LL __int64 int dp[10001]; int sg(int x) { int flag[10001],temp,i; if(dp[x] >= 0) return dp[x]; memset(flag,

hdu2824 The Euler function 筛选法求欧拉函数模板题

//求a , b范围内的所有的欧拉函数 //筛选法求欧拉函数模板题 #include<cstdio> #include<cstring> #include<iostream> using namespace std ; const int maxn = 3000010 ; typedef __int64 ll ; int e[maxn] ; int a ,  b ; void Euler() { int i,j; for (i=1;i<maxn;i++) e[i]

POJ 3528 hdu 3662 三维凸包模板题

POJ 3528题:http://poj.org/problem?id=3528 HDU 3662:http://acm.hdu.edu.cn/showproblem.php?pid=3662 一个是求三维凸包面数,一个是求三维凸包表面积,都是很裸的. 贴代码: #include<stdio.h> #include<algorithm> #include<string.h> #include<math.h> #include<stdlib.h>

HDU 3032 Nim or not Nim?(sg函数博弈)

题目地址:HDU 3032 这题是很好用来练习sg函数打表的一题. 下面是sg函数值打表代码: #include <iostream> #include <cstdio> #include <string> #include <cstring> #include <stdlib.h> #include <math.h> #include <ctype.h> #include <queue> #include &

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[