威佐夫博弈 hdu 2177

威佐夫博弈基本思想:

         两堆物品,可从一堆中取任意,也可从两堆中同时取相同多个。

         该算法存在奇异状态:处在奇异状态必败;

       1.   奇异状态的求法:

          a0=0;b0=0;

          而后,ak=前未出现过的最小自然数;bk=ak+k;

          可得出奇异状态。

        2.   判断是否为奇异状态:

          有一状态(a,b)

               if   a=int((b-a)*(1+sqrt(5))/2);

                则表明处于奇异状态。

        3.  怎样转化成奇异状态:

         ①  a=b时,直接转化成(0,0);

         ②  k=b-a;if (a-ak)=(b-bk); (a-ak)>0,(b-bk)>0

             则转化为(a-(a-ak),b-(b-bk));

         ③   在ak中可以找到与a相等的,或bk中可以找到与a相等的,则转化为(ak,bk);

         以上三种情况中,①与②不会同时出现,③与前两个可以同时出现。

 

Description

有两堆石子,数量任意,可以不同。游戏开始由两个人轮流取石子。游戏规定,每次有两种不同的取法,一是可以在任意的一堆中取走任意多的石子;二是可以在两堆中同时取走相同数量的石子。最后把石子全部取完者为胜者。现在给出初始的两堆石子的数目,如果轮到你先取,假设双方都采取最好的策略,问最后你是胜者还是败者。如果你胜,你第1次怎样取子?
 

Input

输入包含若干行,表示若干种石子的初始情况,其中每一行包含两个非负整数a和b,表示两堆石子的数目,a和b都不大于1,000,000,且a<=b。a=b=0退出。
 

Output

输出也有若干行,如果最后你是败者,则为0,反之,输出1,并输出使你胜的你第1次取石子后剩下的两堆石子的数量x,y,x<=y。如果在任意的一堆中取走石子能胜同时在两堆中同时取走相同数量的石子也能胜,先输出取走相同数量的石子的情况.
 

Sample Input

1 2
5 8
4 7
2 2
0 0
 

Sample Output

0
1
4 7
3 5
0
1
0 0
1 2
 
 
代码:
#include<iostream>
#include<math.h>
using namespace std;
int a[1000000][2];
int temp=1000000;
int temp3=0;
void find(){
     int temp2=0;  
     while(temp2<=temp){
        a[temp3][0]=int(temp3*(1+sqrt(5))/2);
        a[temp3][1]=a[temp3][0]+temp3;
         temp2=a[temp3][1];
         temp3++;
         }
  //   cout<<temp3<<endl;
         }
int main(){
    find();
    int n1,n2;
    while(cin>>n1>>n2){
           if(n1==0&&n2==0)
                 break;
           double tempor=(sqrt(5)+1)/2;
           if(n1==(int)((n2-n1)*tempor)){
                     cout<<"0"<<endl;
                     continue;
                     }
           cout<<"1"<<endl;
           int i=0;
           int sub=n2-n1;
           if(n1==n2){
               cout<<"0 0"<<endl;
               }
                                  
          
          else if(((n1-a[sub][0])==(n2-a[sub][1]))&&((n1-a[sub][0])>0)&&((n2-a[sub][1])>0))
                  cout<<a[sub][0]<<" "<<a[sub][1]<<endl;
      //    cout<<temp3<<endl;
          for(int j=0;j<temp3;j++){
                  if(a[j][0]==n1){
                         cout<<a[j][0]<<" "<<a[j][1]<<endl;
                         }
                  if(a[j][1]==n1){
                         cout<<a[j][0]<<" "<<a[j][1]<<endl;
                         break;
                         }
                         }
                         }
           return 0;
           }
 

 

posted on 2012-07-01 20:54 yumao 阅读(...) 评论(...) 编辑 收藏

导航

公告