题目大意:
就是现在初始的时候有两个数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;
}