您的位置:首页 > 其它

UVa 1378 A Funny Stone Game [博弈论 SG函数]

2017-03-13 16:23 399 查看
A Funny Stone Game

题意:

$n \le 23$堆石子,每次选择$i < j \le k$,从$i$拿走1颗$j,k$各放入一颗,不能取就失败。求先手是否必胜以及第一次取的策略

一开始一直在想游戏怎么会结束...眼残没发现$i<j.....$

然后,解这类组合游戏问题重要的一步是发现独立的子游戏

本题中每个石子是互不影响的

$sg[i]$为一颗在$i$的石子的状态

$sg[i]=mex\{sg[j] \oplus sg[k]\}$

然后就$\oplus$起来就行了

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
using namespace std;
const int N=30;
inline int read(){
char c=getchar();int x=0,f=1;
while(c<'0'||c>'9'){if(c=='-')f=-1; c=getchar();}
while(c>='0'&&c<='9'){x=x*10+c-'0'; c=getchar();}
return x*f;
}
int n,s
;
int sg
,use[N*N];
int main(){
freopen("in","r",stdin);
int cas=0;
while( (n=read()) ){
printf("Game %d: ",++cas);
for(int i=1;i<=n;i++) s[i]=read();
sg
=0;
for(int i=n-1;i>=1;i--){
memset(use,0,sizeof(use));
for(int j=i+1;j<=n;j++)
for(int k=j;k<=n;k++) use[ sg[j]^sg[k] ]=1;
for(int j=0;;j++) if(!use[j]) {sg[i]=j;break;}
}
int ans=0;
for(int i=1;i<=n;i++) if(s[i]&1) ans^=sg[i];
if(ans!=0){
int i,j,k;
for(i=1;i<=n;i++) if(s[i]){
int flag=0; ans^=sg[i];
for(j=i+1;j<=n;j++){
if(ans==0) {k=j,flag=1;break;}
for(k=j+1;k<=n;k++) if( (ans^sg[j]^sg[k])==0 ) {flag=1;break;}
if(flag) break;
}
if(flag) break; ans^=sg[i];
}
printf("%d %d %d\n",i-1,j-1,k-1);
}else puts("-1 -1 -1");
}
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: