HDU 1525 Euclid's Game 博弈

题目大意:

就是现在初始的时候有两个数a, b, 每次可以选择将其中较大的那个数减去较小的那个数的任意倍, 但是得到的数必须不小于0, 谁将其中一个数变成0谁就赢了

初始给定的两个数都是正整数


大致思路:

其实就是一个简单的博弈问题, 我们每次考虑二元组(a, b), 不妨设 a <= b, 那么当(a, b)可以变成 (a - b, b), (a - 2*b, b), ... (a - k*b, b), 但是必须要走到 (a - k*b, b)之后才能改变继续减去b的倍数的局面, 所以每次(a, b)必须要经过变成(a % b, b)这样的状态才能向下走, 而(a, b)有指向(a - b, b)...(a - k*b, b)的边, (a - b, b)有指向(a - 2*b , b)...(a - k*b, b)的边, 那么由于a, b中有一个为0的是P点, 可以知道当状态(a1, b1) -> (a2, b2)之间有k步的时候, 如果(a2, b2)是N点, 当且仅当k == 1时(a1, b1)是P点, 当(a2, b2)是P点时, (a1, b1)一定是N点, 如此向前推理得到初始的(a, b)是N点还是P点即可, 复杂度是对数级别的


代码如下:

Result  :  Accepted     Memory  :  1628 KB     Time  :  31 ms

/*
 * Author: Gatevin
 * Created Time:  2015/4/21 11:06:29
 * File Name: Rin_Tohsaka.cpp
 */
#include<iostream>
#include<sstream>
#include<fstream>
#include<vector>
#include<list>
#include<deque>
#include<queue>
#include<stack>
#include<map>
#include<set>
#include<bitset>
#include<algorithm>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cctype>
#include<cmath>
#include<ctime>
#include<iomanip>
using namespace std;
const double eps(1e-8);
typedef long long lint;
#define foreach(e, x) for(__typeof(x.begin()) e = x.begin(); e != x.end(); ++e)
#define SHOW_MEMORY(x) cout<<sizeof(x)/(1024*1024.)<<"MB"<<endl

lint a, b;
lint t[100];

int main()
{
    while(scanf("%I64d %I64d", &a, &b), a && b)
    {
        if(a < b) swap(a, b);
        int cnt = 0;
        while(a % b)
        {
            t[cnt++] = a / b;//找出多个必经状态之间的步数
            a = a % b;
            if(a < b) swap(a, b);
        }
        bool NP = false;//结尾(a, b)可以到达其中一个为0时, 是N点
        for(int i = cnt - 1; i >= 0; i--)
        {
            if(NP) NP = 0;//N点
            else if(!NP && (t[i] == 1)) NP = 1;//P点
        }
        if(NP) printf("Ollie wins\n");
        else printf("Stan wins\n");
    }
    return 0;
}


你可能感兴趣的:(game,HDU,博弈,Euclids,1525)