pku 2425 A Chess Game (SG)

题意:

给一个由N个点组成的一张有向图,不存在环。点的编号是0~N-1。

然后给出M个棋子所在的位置(点的编号)【一个点上可同时有多个棋子】。

每人每次可移动M个棋子中的一个棋子一步,移动方向是有向边指向的方向。最后无法移动棋子的人输。

思路:

一眼就可看出的裸的SG,直接看代码吧。

代码:

int sg[1005];
vector<int> graph[1005];

int dfs(int x){ // position x
    if(sg[x]!=-1)
        return sg[x];
    int L=graph[x].size();
    if(L==0)
        return sg[x]=0;
    bool vis[1005] = {0};
    rep(i,0,L-1){
        vis[dfs(graph[x][i])] = true;
    }
    for(int i=0;;++i){
        if(!vis[i])
            return sg[x]=i;
    }
}

int n,Xi,a,m,pos;
int main(){
    while(scanf("%d",&n)!=EOF){
        rep(i,0,n-1) graph[i].clear();
        rep(i,0,n-1){
            scanf("%d",&Xi);
            while(Xi--){
                scanf("%d",&a);
                graph[i].push_back(a);
            }
        }
        mem(sg,-1);

        while(scanf("%d",&m),m){
            int ans=0;
            rep(i,1,m){
                scanf("%d",&pos);
                ans=ans^dfs(pos);
            }
            if(!ans)
                puts("LOSE");
            else
                puts("WIN");
        }
    }
}
时间: 2024-12-28 16:48:58

pku 2425 A Chess Game (SG)的相关文章

poj 2425 A Chess Game(SG函数)

A Chess Game Time Limit: 3000MS   Memory Limit: 65536K Total Submissions: 3551   Accepted: 1440 Description Let's design a new chess game. There are N positions to hold M chesses in this game. Multiple chesses can be located in the same position. The

博弈论(SG) hdu5724

百度搜一下博弈论 由感性认识到理性认识的论文 理论铺垫: 1.定义P-position和 N-position:其中P代表Previous,N代表Next.直观的说,上一次move的人有必胜策略的局面是P-position,也就是“先 手必败”,现在轮到move的人有必胜策略的局面是N-position,也就是“先手可保证必胜”. (1).无法进行任何移动的局面(也就是terminal position)是P-position: (2).可以移动到P-position的局面是N-position

HDU 1524 - A Chess Game(sg函数 + dfs)

A Chess Game Problem Description Let's design a new chess game. There are N positions to hold M chesses in this game. Multiple chesses can be located in the same position. The positions are constituted as a topological graph, i.e. there are directed

HDU ACM 1524 A Chess Game-&gt;博弈(SG函数)

题意:一个有向无环图上有n个顶点,每一个顶点都可以放一个棋子或不放,有两个人,每次根据这个图只能将任意一颗棋子移动一步,如果到某一步玩家不能移动时,那么这个人就输. 分析: 1.有向无环图的博弈,dfs把所有顶点的SG值都计算出来,然后对每个棋子的SG值进行异或运算,为0就是先手必败,否则就是先手必胜. 2.如果某个人移动后,所有棋子都在出度为0的顶点,那么他必败. SG函数简介: a.对于给定的有向无环图,定义图中每个顶点的Sprague-Grundy函数g如下:g(x) = mex{ g(y

POJ 2425 A Chess Game 博弈论 sg函数

http://poj.org/problem?id=2425 典型的sg函数,建图搜sg函数预处理之后直接求每次游戏的异或和.仍然是因为看不懂题目卡了好久. 这道题大概有两个坑, 1.是搜索的时候vis数组应该在函数内声明(似乎这是我经常在搜索里犯的错误,为了省一点空间整道题都写错了): 2.是n个点的有向无环图边数上限是n^2(re了好久QAQ). 在漫长的查资料过程之后终于大概搞懂了sg函数的原理,愉快.下一篇大概会写一个小结. 代码 1 #include<cstdio> 2 #inclu

hdu 1848 Fibonacci again and again (SG)

题意: 3堆石头,个数分别是m,n,p. 两个轮流走,每走一步可以选择任意一堆石子,然后取走f个.f只能是菲波那契中的数(即1,2,3,5,8.....) 取光所有石子的人胜. 判断先手胜还是后手胜. 思路: 简单SG.看代码. 代码: int sg[1005]; int dfs(int x){ if(sg[x]!=-1) return sg[x]; bool vis[1005]={0}; int t1=1,t2=2; if(x>=t1) vis[dfs(x-t1)]=true; while(x

poj 3537 Crosses and Crosses (SG)

题意: 1 × n 个格子,每人每次选一个格子打上叉(不得重复),如果一个人画完叉后出现了连续的三个叉,则此人胜. 给n,判断先手胜还是先手败. 思路: 假设选择画叉的位置是i,则对方只能在前[1,i-3]中或[i+3,n]中选择画叉.子问题出现. 根据SG的定义,即可求出SG(N).看代码. 代码: int sg[2005]; int n; int dfs(int n){ if(n<0) return 0; if(sg[n]!=-1) return sg[n]; bool g[2005] =

hdu 1847 Good Luck in CET-4 Everybody!(sg)

Good Luck in CET-4 Everybody! Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 9703    Accepted Submission(s): 6286 Problem Description 大学英语四级考试就要来临了,你是不是在紧张的复习?也许紧张得连短学期的ACM都没工夫练习了,反正我知道的Kiki和Ci

POJ 2425 A Chess Game#树形SG

http://poj.org/problem?id=2425 #include<iostream> #include<cstdio> #include<cstring> #include<algorithm> using namespace std; struct node { int to,next; }e[10000010]; int head[1010],Ecou; int sg[1010]; void add_edge(int u,int v) {