hdu 2177 取(2堆)石子游戏

天资愚笨啊,网上的一大堆没看懂。。。。。。

总结百科的方法为:

1.a==b

同时减去a 得到0,0

2.a==a_k      b>b_k

b -(b-b_k)

3.a==a_k     b<b_k

同时拿走a_k-a_(b-a_k)

得到 a_(b-a_k)    a_(b-a_k) + b-a_k

4.a>a_k       b==b_k

从a中拿走 a-a_k

5.a<a_k       b==b_k

5.1 a==a_ j   (j<k)

b-(b-b_ j)

得到 a_ j    b_ j

5.2 a==b_ j   (j<k)

b-(b-a_ j)

得到 b_ j   a_ j

反正我是没搞懂!!o(╯□╰)o。。。。

我的方法就是 穷举了:

第一个分支,从两堆物品中同时取出相同数量的物品

第二个分支,只从一堆物品中取物品

代码如下:

#include<stdio.h>
#include<math.h>
int main()
{
  int a,b,k,temp,n,i,aa,bb,x,y;
  double m;

  while(~scanf("%d%d",&a,&b)&&a+b)
  {
    if(a>b)
    {
      temp=a;
      a=b;
      b=temp;
    }
    k=b-a;
    m=(1+sqrt(5.0))/2.0;
    n=(int)k*m;
    if(a==n)
      printf("0\n");
    else
     {
         printf("1\n");
         for(i=1;i<=a;i++)
         {
           aa=a-i;
           bb=b-i;
           if(n==aa)
           {
              printf("%d %d\n",aa,bb);
              break;
           }
         }
         for(i=b;i>=0;i--)
         {
            x=a;y=i;
            if(x>y)
            {
              temp=y;
              y=x;
              x=temp;
            }
            k=y-x;
            n=(int)k*m;
            if(x==n)
             {
               printf("%d %d\n",x,y);
               break;
             }
         }

     }
  }
  return 0;
}

hdu 2177 取(2堆)石子游戏,布布扣,bubuko.com

时间: 2024-12-28 01:33:18

hdu 2177 取(2堆)石子游戏的相关文章

HDU 2177 取(2堆)石子游戏 威佐夫博弈

题目来源:HDU 2177 取(2堆)石子游戏 题意:中文 思路:判断是否是必败态就不说了 做过hdu1527就知道了 现在如果不是必败态 输出下一步所有的必败态 题目要求先输出两堆都取的方案 首先 a = b 直接2堆取完 a != b 因为bi = ai+i 现在知道ak 和 bk 那么 k = bk-ak 得到k 求出 aj 和 bj 如果ak-aj == bk-bj && ak-aj > 0(aj, bj)是必败态 输出aj bj 然后是只取一堆的情况 假设a不变 求出对应的

hdu 2177 取(2堆)石子游戏 博弈论,,威佐夫博奕 ,提高题,不过测试数据奇水无比

取(2堆)石子游戏 Time Limit: 3000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submission(s): 1291    Accepted Submission(s): 764 Problem Description 有两堆石子,数量任意,可以不同.游戏开始由两个人轮流取石子.游戏规定,每次有两种不同的取法,一是可以在任意的一堆中取走任意多的石子:二是可以在两堆中同时取走相

hdu 2177 取(2堆)石子游戏 (威佐夫博奕)

//,在威佐夫博奕的基础上新增加了一条要求:就是如果在赢得条件下,输出第一步怎么走. # include <stdio.h> # include <algorithm> # include <iostream> # include <math.h> # include <string.h> using namespace std; int main() { int a,b,i,k; while(~scanf("%d%d",&a

hdu 2177 取(2堆)石子游戏(威佐夫博奕)

题目链接:hdu 2177 这题不是普通的 Nim 博弈,我想它应该是另一种博弈吧,于是便推 sg 函数打了个 20*20 的表来看,为了方便看一些,我用颜色作了标记,打表代码如下: 1 #include<cstdio> 2 #include<cstring> 3 #include<string> 4 #include<map> 5 #include<algorithm> 6 #include<windows.h> 7 using n

HDU 2176 取(m堆)石子游戏 博弈

取(m堆)石子游戏 Time Limit: 3000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 3598    Accepted Submission(s): 2151 Problem Description m堆石子,两人轮流取.只能在1堆中取.取完者胜.先取者负输出No.先取者胜输出Yes,然后输出怎样取子.例如5堆 5,7,8,9,10先取者胜,先取者第1次取

HDU 2176 取(m堆)石子游戏 (Nim博弈)

取(m堆)石子游戏 Time Limit: 3000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 1937    Accepted Submission(s): 1115 Problem Description m堆石子,两人轮流取.只能在1堆中取.取完者胜.先取者负输出No.先取者胜输出Yes,然后输出怎样取子.例如5堆 5,7,8,9,10先取者胜,先取者第1次取

HDU 2176 取(m堆)石子游戏 (尼姆博奕)

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2176 m堆石子,两人轮流取.只能在1堆中取.取完者胜.先取者负输出No.先取者胜输出Yes,然后输出怎样取子.例如5堆 5,7,8,9,10先取者胜,先取者第1次取时可以从有8个的那一堆取走7个剩下1个,也可以从有9个的中那一堆取走9个剩下0个,也可以从有10个的中那一堆取走7个剩下3个. Input输入有多组.每组第1行是m,m<=200000. 后面m个非零正整数.m=0退出. Output先取

HDU 2176 取(m堆)石子游戏 &amp;&amp; HDU1850 Being a Good Boy in Spring Festivaly

HDU2176题意: m堆石子,两人轮流取.只能在1堆中取.取完者胜.先取者负输出No.先取者胜输出Yes,然后输出怎样取子. 通过 SG定理 我们可以知道每一个数的SG值,等于这个数到达不了的前面数中的最小值.本题题意和尼姆博弈一样,即可以在一堆中任意个石子,所以也就是说每个数都可以到达前面经过的每一个数,所以每一个数的SG值就是它本身.又因为有好多堆石子,所以可以看作多个一堆石子的游戏,我们可以让n代表每一堆石子的数量,那么让所有堆的SG(n)相互异或得到的结果就是答案(这里只是用SG定义来

hdu 2176 取(m堆)石子游戏 (裸Nim)

题意: m堆石头,每堆石头个数:a[1]....a[m]. 每次只能在一堆里取,至少取一个. 最后没石子取者负. 先取者负输出NO,先取胜胜输出YES,然后输出先取者第1次取子的所有方法.如果从有a个石子的堆中取若干个后剩下b个后会胜就输出a b 思路: 裸的NIM. 单看一堆石子,没有石头sg[0]=0,一个石头sg[1]=1,....n个石头sg[n]=n. 故SG[a[1],a[2]...a[m]] = sg[a[1]]^...^sg[a[m]] = a[1]^...^a[m] SG=0