每天一道博弈论之“高手过招”(非口胡)

   题目大意:

  给你一个n行20列的棋盘,上面有一些棋子

  每个棋子只能往右边第一个空格处移动,无棋子可移动即判定为输

  现在给你棋盘大小和棋子分布,问先手是否必胜

 

    题解:

   可以看出每行是独立的,也就是说只要求出每行的SG函数值,在亦或一下就可以了(SG定理^_^)

   又由于每行长度一样,所以可以预处理每个状态的SG值

   把每个的状态对应成一个二进制数,就能方便地求出SG值了qaq

 

 题目来源:https://www.luogu.org/problemnew/show/P2575

 

(今天有代码哦qwq)

#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
const int N = 1000 + 10 ;

inline int read() {
    int k = 0 , f = 1 ; char c = getchar() ;
    for( ; !isdigit(c) ; c = getchar())
      if(c == '-') f = -1 ;
    for( ; isdigit(c) ; c = getchar())
      k = k*10 + c-'0' ;
    return k*f ;
}

int sg[1<<20] ;
bool vis[28] ;

inline void solve(int x) {
    memset(vis,0,sizeof(vis)) ;
    int nxt = 0, ww = 1 ;
    for(int i=1;i<=20;i++) {
        if(x&ww) {
             if(nxt) vis[sg[x^ww^(1<<(nxt-1))]] = 1 ;
        } else nxt = i ;
        ww <<= 1 ;
    }
    for(int i=0; ;i++) {
        if(!vis[i]) {
            sg[x] = i ; return ;
        } 
    }
}

int main() {
    int t = read(), mm = (1<<20) ;
    for(int i=0;i<mm;i++) solve(i) ;
    int n, m, ans, x, tmp ;
    while(t--) {
        ans = 0 ;
        n = read() ;
        for(int i=1;i<=n;i++) {
            m = read() ; tmp = 0 ;
            for(int j=1;j<=m;j++) {
                x = read() ;
                tmp += (1<<(20-x)) ;
            }
            ans ^= sg[tmp] ;
        }
        if(ans) printf("YES\n") ;
        else printf("NO\n") ;
    }
    return 0 ;
}

 

 

 

 

 

posted @ 2018-03-01 22:33  zubizakeli  阅读(184)  评论(0编辑  收藏  举报