/** \brief hdu 1222 Wolf and Rabbit
*
* \param date 2014/7/22
* \param state AC
* \return
*
*/
#include <iostream>
#include <fstream>
using namespace std;
int gcd(int x,int y)
{
while(x!=y)
{
if(x>y)x=x-y;
else y=y-x;
}
return x;
}
int gcd2(int x,int y)
{
if(x<y)
return gcd(y,x);
if(y==0)
return x;
else return gcd(x-y,y);
}
//AC
int gcd3(int m,int n)
{
while(n%m)
{
int temp=n%m;
n=m;
m=temp;
}
return m;
}
int main()
{
//cout << "Hello world!" << endl;
//freopen("input.txt","r",stdin);
int P;
cin>>P;
for(int i=0;i<P;i++)
{
int m,n;
cin>>m>>n;
int flag=gcd3(m,n);
if(flag==1)cout<<"NO"<<endl;
else cout<<"YES"<<endl;
}
return 0;
}
理解:
该题是要求判断出是否能通过约瑟夫环的数数方式将所有的点都数出来,结论很简单,就是判定总人数和跳跃人数是否互质,如果最大公约数是一的话,那么输出NO,否则输出YES。证明如下:
设在N个点的圈中,按每数M个标记一下,那么能够把所有的点遍历到即要满足在循环完一圈后,新的起点一定要在1-M中的任何一点出现一次,这个不难理解吧。那么问题就转化为能否遍历1-M中所有的点,依次如此进行下去,当发现不能满足小区间的要求时,那么其父层亦不能满足要求。那么每次起点的偏移量是否有规律呢?答案是有的,以初始的N和M为例,每次的偏移量是 N%M( 假设N> M反之交换位置 ),也即取余的一个过程,那么接下来的新的N就是M,M也就变成了 N%M, 即 N2<-- M1;