【博弈】poj 2960 S-Nim

虽然题目有一点变形,不过本质还是用SG函数,nim(n)=mex{nim(n-s[i])},想清楚这一点,问题就比较容易解决了。

分析:
 1.可将问题转化为n个子问题,每个子问题分别为:
   从一堆x颗石子中取石子,每次可取的石子数为集合S(k)中的一个数
 2.分析(1)中的每个子问题,
   易得:SG(x) = mex(SG[x-s[i]])(0<i<k-1); 
 3.后面就是SG函数的应用,根据Sprague-Grundy Therem:g(G)=g(G1)^g(G2)^g(G3)^...^g(Gn)
   即游戏的和的SG函数值是它的所有子游戏的SG函数值的异或,即
   SG(G) = SG(x1)^SG(x2)^...^SG(xn),故若SG(G)=0那么必输

#include<iostream>
using namespace std;
int main(){
	int t,m,number,goal[10005],s[105],tmp,mark[10005];
	while(cin>>t&&t){
		for(int i=0;i<t;i++)
			cin>>s[i];
		goal[0]=0;
		for(int i=0;i<10005;i++)
			mark[i]=9999999;
		for(int i=1;i<=10000;i++){			
			for(int j=0;j<t;j++){
				if(i>=s[j]){
					mark[goal[i-s[j]]]=i;
				}
			}
			goal[i]=0;
			while(mark[goal[i]]==i)
				goal[i]++;
		}		
		cin>>m;
		for(int i=0;i<m;i++){
			cin>>number;
			int sum=0;
			for(int j=0;j<number;j++){
				cin>>tmp;				
				sum=sum^goal[tmp];
			}
			if(sum==0)
				cout<<"L";
			else cout<<"W";
		}
		cout<<endl;
	}
	
	return 0;
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值