nyoj 998
nyoj 998 点击这里打开题目链接
给你一个数N,使得在1~N之间能够找到x使得x满足gcd( x , N ) >= M,
求解gcd(x,N)的和
思路:一开始想到暴力法做,超时 ,后来借鉴学长经验AC:
大致思路: 用欧拉函数求 ,euler(n) 表示 1到n与n互质的数的个数, 如果n能够被 i 整除 ,
则euler(n/i)等价于gcd(x,N)==i 的个数 (x是从1到n的所有数字)。
所以,你会想到一个for循环,因为gcd(x,N)的值为1到n 枚举 gcd为M到n,再来求和? 时间复杂度也会很高,
所以并不需要全部一一枚举
比如gcd(x,N) N为 10 , 它们的gcd 只有 1,10 2,5
N为15 它们的gcd 有 1,15, 3,5
所以它们的gcd 只需要知道1 到 根号n 而另外一个gcd 为 n/i ;
for(i=1;i*i<=n;i++)
{
if(n%i==0)
{
if(i>=m) sum+=i*euler(n/i);
if(i*i!=n&&n/i>=m) sum+=(n/i)*euler(i);
}
}注意: 第二个if i *i!=n 因为在第一个if里面已经加过了
下面贴个代码
#include <stdio.h>
#include <math.h>
long long euler(long long n)
{
long long i,sum=n,temp=n;
for(i=2;i*i<=temp;i++)
{
if(n%i==0)
{
sum=sum/i*(i-1);
while(n%i==0)
n/=i;
}
}
if(n>1)
sum=sum/n*(n-1);
return sum;
}
int main()
{
long long m,n;
while((scanf("%lld %lld",&n,&m))!=EOF)
{
long long sum=0,i;
for(i=1;i*i<=n;i++)
{
if(n%i==0)
{
if(i>=m) sum+=i*euler(n/i);
if(i*i!=n&&n/i>=m) sum+=(n/i)*euler(i);
}
}
printf("%lld\n",sum);
}
return 0;
}

浙公网安备 33010602011771号