几种博弈问题及算法

巴什博弈

巴什博弈:只有一堆n个物品,两个人轮流从这堆物品中取物,规定每次至少取一个,最多取m个。最后取光者得胜。
显然,如果n=m+1,那么由于一次最多只能取m个,所以,无论先取者拿走多少个,后取者都能够一次拿走剩余的物品,后者取胜。因此我们发现了如何取胜的法则:如果n=(m+1)r+s,(r为任意自然数,s≤m),那么先取者要拿走s个物品,如果后取者拿走k(≤m)个,那么先取者再拿走m+1-k个,结果剩下(m+1)(r-1)个,以后保持这样的取法,那么先取者肯定获胜。总之,要保持给对手留下(m+1)的倍数,就能最后获胜。

题目链接:Brave Game

#include<stdio.h>
int main()
{
	int t;
	scanf("%d",&t);
	while(t--)
	{
		int n,m;
		scanf("%d%d",&n,&m);
		if(n%(m+1))
		printf("first\n");
		else
		printf("second\n");
	}
 } 

题目链接:Public Sale

#include<stdio.h>
int main()
{
	int n,i,m;
	while(~scanf("%d%d",&m,&n))
	{
		if(m%(n+1)==0)
		printf("none\n");
		else
		{
			if(m<=n)
			{
				if(m==n)//如果m,n一样,那么第一次出价就一定要直接买下来 
				printf("%d\n",m);
				else
				{
					for(i=m;i<=n-1;i++)//或者就是比m大(包括m)到n(包括n)的都可以 
					printf("%d ",i);
					printf("%d\n",n);
				}
			}
			else
			printf("%d\n",m%(n+1));//如果m比n大,为了保证一定赢,就要出m%(n+1) 
		}
	}
}

威佐夫博弈

有两堆石子,两个玩家可以(1)从两堆石子中分别取相同数目的石子(2)从一堆石子中取任意数目的石子,求解先手是否能赢。
假设面对的局势是(a,b),在这里插入图片描述
如果成立,则此局面为奇异局势,先手必败。

题目链接:取石子游戏

#include<math.h>
#include<stdio.h>
#include<algorithm>
using namespace std;
int main()
{
	int n,m,a,b,c;
	double k,l;
	while(~scanf("%d%d",&n,&m))
	{
		a=min(n,m);
		b=max(n,m);
		k=(sqrt(5.0)+1)/2;
		l=(double)(b-a);
		c=int(k*l);
		if(c==a)
		printf("0\n");
		else
		printf("1\n");
	}
}

尼姆博弈

尼姆博弈可以说是巴什博弈的加强版,即有N堆石子,可以从一堆石子中拿走任意个,先拿完者胜
当异或值为0,说明先取者输,直接输出No即可,当异或值不为0,说明他取石子的数量能保证它必胜,异或值的结果即为剩下的石子的数量

题目链接:Being a Good Boy in Spring Festival

#include<stdio.h>
const int maxn=4e5+5;
int a[maxn];
int main()
{
	int i,j,n,m,k,sum;
	while(~scanf("%d",&n)&&n)
	{
		m=0;
		sum=0;
		for(i=0;i<n;i++)
		{
			scanf("%d",&a[i]);
			m=m^a[i];
		}
		if(!m)
		printf("0\n");
		else
		{
			for(i=0;i<n;i++)
			if(a[i]>(m^a[i]))//注意'^'的优先级小于'>'
			sum++;
			printf("%d\n",sum);
		}
	}
}

题目链接:Matches Game

#include<stdio.h>
const int maxn=4e5+5;
int a[maxn];
int main()
{
	int i,j,n,m,k;
	while(~scanf("%d",&n))
	{
		m=0;
		for(i=0;i<n;i++)
		{
			scanf("%d",&a[i]);
			m=m^a[i];
		}
		if(!m)
		printf("No\n");
		else
		{
			printf("Yes\n");
		}
	}
}

题目链接:取(m堆)石子游戏

#include<stdio.h>
const int maxn=4e5+5;
int a[maxn];
int main()
{
	int i,j,n,m,k;
	while(~scanf("%d",&n)&&n)
	{
		m=0;
		for(i=0;i<n;i++)
		{
			scanf("%d",&a[i]);
			m=m^a[i];
		}
		if(!m)
		printf("No\n");
		else
		{
			printf("Yes\n");
			for(i=0;i<n;i++)
			{
				k=m^a[i];
				if(k<a[i])
				printf("%d %d\n",a[i],k);
			}
		}
	}
}

斐波那契博弈

1堆石子有n个(n>1),两人轮流取.先取者第1次可以取任意多个,但不能全部取完.以后每次取的石子数不能超过上次取子数的2倍,先取完者胜。

结论:当n为Fibonacci数的时候,必败。

题目链接:取石子游戏

#include<stdio.h>
int main()
{
	int n,i,f[100];
	f[0]=2;
	f[1]=3;
	for(i=2;i<100;i++)
	f[i]=f[i-1]+f[i-2];
	while(~scanf("%d",&n)&&n)
	{
		int flag=0;
		for(i=0;i<100;i++)
		{
			if(f[i]==n)
			{
				flag=1;
				printf("Second win\n");
				break;
			}
		}
		if(!flag)
		printf("First win\n");
	}
}

环形博弈

n个石子围成一个环,每次可取走1个或相邻的两个,注意若两个石子之间的石子被取走,这两个石子仍然是不相邻的。

对于n=1和n=2的情况,先手必胜。

当n=3时,显然先手必败。
当n>3时,我们可以对这个环画一个对称轴,无论先手怎么取,后手只要让这个环呈对称状就能赢。

题目链接:A Funny Game

#include<stdio.h> 
int main()
{
    int n;
    while(~scanf("%d",&n)&&n)
    {
        if(n==1||n==2)
        printf("Alice\n");
        else
        printf("Bob\n");
    }
}

题目链接:Coin Game

当一次能取完时,先手必胜。当k=1,n是奇数时,先手必胜。其他时,后手必胜。

#include<stdio.h> 
int main()
{
    int t,l=1,n,k;
    scanf("%d",&t);
    while(t--)
    {
        scanf("%d%d",&n,&k);
        printf("Case %d: ",l++);
        if(n<=k)
        {
        	printf("first\n");
        	continue;
		}
		if(k==1&&n%2==1)
		{
			printf("first\n");
			continue;
		}
		printf("second\n");
    }
}
  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值