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