nim取子游戏 阶梯博弈

nim取子游戏

问题描述:

有k堆石子,两个人I,II分别轮流取子。

一次可以从一堆中至少取一个,至多取完。

最后一个取完的胜利。


问I先手,是否有必胜的策略。


==========================================================================


先考虑个简单的2堆石子。

如果两个石子数量不同,I先取使得两堆数量相同。然后I就模仿II的取法在另一堆取。

这样I就必胜。

若相同,那么II模仿I的取法,II必胜。


如果是k堆呢?


我们来考虑每堆石子的表示方法,用二进制表示之。


因为二进制可以表示任何数。


和刚才两堆类似。


如果所有堆的相同二进制位的1的个数都是偶数,那么肯定II必胜。


如果不满足上面这种情况,那么I必胜。

因为I可以通过一次取子,让其满足此种情况。


k = 4

分别是 7 9 12  15


二进制表示


0111

1001

1100

1111


每位的1的数量

3323,显然是不满足的,所以I必胜。


I的策略就是在15个的上面取

1101个。。。(1101)2=13个

那么就变成了

0111

1001

1100

0010


那样,这时轮到II取,这时无论II怎么取,我们都可以一直让此状态保持平衡。


那么这样到最后还是I赢。


==========================================================================

nim取子游戏就这样了。。呵呵~~


编程也很简单的。

因为每一位个相同位1的个数是偶数,那么我把所有数异或之后肯定就得到的是0。。。

如果非0那么I赢。


阶梯博弈

首先介绍一下阶梯博弈:

    游戏开始时有许多硬币任意分布在楼梯上,共n阶楼梯从地面由下向上编号为0n。游戏者在每次操作时可以将楼梯j(1<=j<=n)上的任意多但至少一个硬币移动到楼梯j-1上。游戏者轮流操作,将最后一枚硬币移至地上的人获胜。

    先证明:奇数台阶的P态一定是游戏的p态。

    首先我们看终止态的前一个状态,一定是第一个台阶上有若干个硬币。我们不妨将奇数的台阶看作是石子堆,放到偶数的台阶看作是拿走。设SG(奇数台阶)=0,如果对方移动奇数台阶的硬币(sg=0,所以不可能是最后一步),那么一定能够找到另外一个台阶做相同的操作;如果对方移动偶数台阶的硬币到某个奇数台阶(也不会是最后一步),我们就可以把这些硬币继续往下移,变成奇数上的硬币,即证。

hdu1907

//hdu1907  三堆取子,最后取完者输 
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
#define MAX 50
int main(){
	int t,n,cont[MAX],ans,k;
	scanf("%d",&t);
	while(t--){
		scanf("%d",&n);
		k=0,ans=0;
		for(int i=0;i<n;i++){
			scanf("%d",&cont[i]);
			if(cont[i]==1){
				k++;
			}
			ans^=cont[i];
		}
		if(k==n){
			if(k&1){
				printf("Brother\n");
			}
			else{
				printf("John\n");
			}
		}else
			if(ans){
				printf("John\n");
			}
			else
			{
				printf("Brother\n");
			}
	}
	return 0;
} 



  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值