POJ 2068 Nim (dp博弈)

题意:

共n轮,s个石头和两队人,两队人轮流拿,第i轮两队分别只能拿1~M[(2*i-1)%(2*n)]和1~M[(2*i)%(2*n)]个石头,拿到最后那个石头的队输;

就拿最后一组样例来说,循环中共3轮,共97个石头,第一轮两队分别拿不超过8个和7个石头,第二轮6个和5个,第三轮4个和3个,然后又是8个和7个......

Input

n S M[1]  M[2] . . . M[2*n]

1 <= n <= 10, 1 <= Mi <= 16, and 1 <= S < 2^13. 

Output

赢输出1,输输出0

Sample Input

1 101 4 4
1 100 4 4
3 97 8 7 6 5 4 3
0

Sample Output

0
1
1

必胜态有至少一种方法转移到必败态

必败态只能转移到必胜态

然后就可以dp了

dp[i][j]表示第i次还剩下j个石头时,1为胜,0为败

然后dfs一下,记忆化一下已经访问过的就好了

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

int a[20],dp[20][9000],n;

int dfs(int k,int left)
{
	if(dp[k][left]!=-1)  return dp[k][left];
	if(left==0)  return dp[k][left]=1;  //当剩下石头数量是0时,是胜的
	for(int i=1;i<=a[k]&&i<=left;i++)
	{
		if(!dfs((k+1)%(2*n),left-i))  return dp[k][left]=1;  //如果可以转移到必败态,那么就是必胜态
	}
	return dp[k][left]=0;
}

int main()
{
	int s;
	while(~scanf("%d",&n)&&n)
	{
		scanf("%d",&s);
		for(int i=0;i<2*n;i++)  scanf("%d",&a[i]);
		memset(dp,-1,sizeof(dp));
		printf("%d\n",dfs(0,s));
	}
	return 0;
}

  

时间: 2024-10-03 22:37:58

POJ 2068 Nim (dp博弈)的相关文章

poj 2068 Nim(博弈dp)

Nim Time Limit: 1000MS   Memory Limit: 30000K Total Submissions: 1403   Accepted: 791 Description Let's play a traditional game Nim. You and I are seated across a table and we have a hundred stones on the table (we know the number of stones exactly).

POJ 2068 Nim#双人dp博弈

http://poj.org/problem?id=2068 #include<iostream> #include<cstdio> #include<cstring> #include<algorithm> using namespace std; int dp[25][(1<<13)+5];//dp[i][j]表示轮到第i个人取时,剩j个石头 int n,s,m[25]; int DFS(int pos,int remain) { if(dp

POJ 2068 Nim

链接: http://poj.org/problem?id=2068 题意: 传统的Nim游戏由两名玩家进行,在一堆石头中,双方轮流取走任意合法数量块石头,取走最后一块石头的玩家落败. 多人Nim游戏将参赛人数拓展至两个队伍,每支队伍有n名队员交错入座,单次分别能最多取走Mi块石头,取走S块石头中的最后一块的队伍失败, 求第一支队伍是否有必胜策略? 题解: dp[i][j]表示第i个人取,还有j块石头 . 当j为0的时候,没有石头,这时候是胜,为1. 后继中有必败态的为必胜态. 代码: 31 i

poj 2975 Nim(博弈)

Nim Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 5232   Accepted: 2444 Description Nim is a 2-player game featuring several piles of stones. Players alternate turns, and on his/her turn, a player’s move consists of removing one or mor

【POJ2068】Nim DP博弈

题意: 多组数据 两人轮流操作,n轮一循环,给出总石子数和这n轮每次两人能取的石子上限(下限为1). 取到最后一颗者输. 比如 3 97 8 7 6 5 4 3 表示一循环有三轮, 可取的个数为: 第一轮 先手8 后手7 第二轮 先手6 后手5 第三轮 先手4 后手3 然后三轮每取完的话就进入下次循环. 数据范围自己看去吧. 题解: DP就好. 博弈性质:如果当前状态对手怎么走都败,此状态就是胜,不然就是负. 记忆化搜索一下水过. 代码: #include <cstdio> #include

POJ 2068 Nim 组合游戏

题目大意:有一堆石子,两伙人,围在一起坐,坐的顺序是ABABABAB....每一个人最多能取a[i]个石子,取走最后一个石子的就输了.问谁能赢. 思路:朴素的组合游戏判定问题,这个题给了数据范围,可以进行记忆化搜索.f[i][j]为还剩下i个石子,到了第j个人的时候的状态,然后记忆化一下. CODE: #include <cstdio> #include <cstring> #include <iostream> #include <algorithm> #

poj 3342(树形dp)

题意:在一个公司中要举办一个聚会,每一个员工有一个奉献值.为了和谐规定直接上下级不能一起出席.让你找出奉献值之和最大为多少. 思路:dp[v][1]表示当前结点选,能获得的最大奉献值,dp[v][0]表示当前节点不选能获得的最大奉献值.状态转移: dp[v][0] = max(dp[v][0], ∑max(dp[x][1], dp[x][0]))x为直接儿子 dp[v][1] = max(dp[v][1], ∑dp[x][0] + vex[v]) 最后答案是max(dp[root][0], dp

poj 2151 概率dp

//poj 2151 概率dp 1 #include "iostream" 2 #include "cstdio" 3 #include "cstring" 4 #include "algorithm" 5 using namespace std; 6 double dp[33][33]; 7 int M, T, N; //problem, team, least 8 double p[1010][33]; 9 int mai

POJ 2250 Compromise (DP,最长公共子序列)

Compromise Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 6440 Accepted: 2882 Special Judge Description In a few months the European Currency Union will become a reality. However, to join the club, the Maastricht criteria must be fulfille