巴什博弈
巴什博弈:只有一堆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");
}
}