Description
FGD正在破解一段密码,他需要回答很多类似的问题:对于给定的整数a,b和d,有多少正整数对x,y,满足x<=a,y<=b,并且gcd(x,y)=d。作为FGD的同学,FGD希望得到你的帮助。
Input
第一行包含一个正整数n,表示一共有n组询问。(1<=n<= 50000)接下来n行,每行表示一个询问,每行三个正整数,分别为a,b,d。(1<=d<=a,b<=50000)
Output
对于每组询问,输出到输出文件zap.out一个正整数,表示满足条件的整数对数。
Sample Input
2
4 5 2
6 4 3
4 5 2
6 4 3
Sample Output
3
2
2
HINT
对于第一组询问,满足条件的整数对有(2,2),(2,4),(4,2)。对于第二组询问,满足条件的整数对有(6,3),(3,3)。
莫比乌斯反演。。记得要用int计算不能用longlong不然会TLE
<pre name="code" class="cpp">#include<cstdio>
#include<string>
#include<cstring>
using namespace std;
int mu[50001],prime[50001];
bool check[50001];
int tot;
int sum[50001];
inline int min(int x,int y)
{
if(x<y)
return x;
return y;
}
inline void findmu()
{
memset(check,false,sizeof(check));
mu[1]=1;
int i,j;
for(i=2;i<=50000;i++)
{
if(!check[i])
{
tot++;
prime[tot]=i;
mu[i]=-1;
}
for(j=1;j<=tot;j++)
{
if(i*prime[j]>50000)
break;
check[i*prime[j]]=true;
if(i%prime[j]==0)
{
mu[i*prime[j]]=0;
break;
}
else
mu[i*prime[j]]=-mu[i];
}
}
for(i=1;i<=50000;i++)
sum[i]=sum[i-1]+mu[i];
}
int main()
{
// freopen("zap.in","r",stdin);
// freopen("zap.out","w",stdout);
findmu();
int b,d,k;
int T;
scanf("%d",&T);
while(T!=0)
{
T--;
scanf("%d%d%d",&b,&d,&k);
if(b>d)
{
int tt=b;
b=d;
d=tt;
}
int ans=0;
int i;
int la=0;
b=b/k;
d=d/k;
for(i=1;i<=b;i=la+1)
{
la=min(b/(b/i),d/(d/i));
ans+=(b/i)*(d/i)*(sum[la]-sum[i-1]);
}
printf("%d\n",ans);
}
return 0;
}