“科大讯飞杯”第十七届同济大学程序设计预选赛 -------- A 张老师和菜哭武的游戏

题目描述 

天才程序员菜哭武和张老师有一天到一个城市旅游,旅途中菜哭武觉得无聊就想和张老师玩一个游戏。菜哭武有n个石子,每个石子都标有1到n之间到数,且各不相同,一开始他们会随机从这堆石子选一个石子放置到一个集合中,张老师选的数是a,菜哭武选的是b(a和b不相同)。接下来菜哭武和张老师轮流按照如下规则拿走一个石子:当石子x能被拿走时,当且仅当集合存在y和z,满足x等于y+z或者y-z,当x被拿走时,把它放到集合中。谁完成最后一轮操作时,谁获胜。张老师总是先手,于是张老师就好奇当决定好a和b时,他是否总是能获胜,你能帮助一下张老师吗?

输入描述:

第一行一个整数T(1≤T≤500),表示共有T组测试数据。
对于每组测试数据,第一行三个整数n(2≤n≤20000)、a和b(1≤a,b≤n, a≠b)。

输出描述:

若张老师能获胜输出Yes,反之No。

示例1

输入

16
2 1 2
3 1 3
67 1 2
100 1 2
8 6 8
9 6 8
10 6 8
11 6 8
12 6 8
13 6 8
14 6 8
15 6 8
16 6 8
1314 6 8
1994 1 13
1994 7 12

输出

No
Yes
Yes
No
No
No
Yes
Yes
No
No
Yes
Yes
No
Yes
No
No

思路: 显然在1~n的这n个数中的某个数A要抽走,要满足A=x*a+y*b,那么显然A必须要是gcd(a,b)的倍数,所以判断n/gcd(a,b)奇偶即可。

比赛的时候没有看出来 使用 欧几里得算法。没感觉 这个算法和这个题沾上边。。(还是自己太太菜了。。),比赛结束后问过学姐才看出来为啥是要使用欧几里得。

为了防止下次遇见还不知道如何 使得与欧几里得挂钩,所以。。。

代入个例子思考一下:比方说一共n个点  最开始取的2  4
接下来仅仅能够取的数 是 6 如果取到6 集合变成 2 4 6
再接下来能取的数是 8 10(意思是抽到这些数才是能放进集合的,别的都不能)你看 8和10 是不是x*2+y*4

代码:

#include
#include
#include
#include
using namespace std;
int gcd(int a,int b)
{
    if(b==0)
        return a;
    return gcd(b,a%b);
}
int main()
{
    int t;
    scanf("%d",&t);
    while(t--)
    {
        int n,a,b;
        scanf("%d%d%d",&n,&a,&b);
        int sum=gcd(a,b);
        if((n/sum)%2)
            printf("Yes\n");
        else
            printf("No\n");
    }
    return 0;
}


 

你可能感兴趣的:(数论)