BZOJ 2818 GCD 欧拉函数

题目:
http://www.lydsy.com/JudgeOnline/problem.php?id=2818
题意:
Description

给定整数N,求1<=x,y<=N且Gcd(x,y)为素数的
数对(x,y)有多少对.

Input

一个整数N
Output

如题
Sample Input
4

Sample Output
4
HINT

hint

对于样例(2,2),(2,4),(3,3),(4,2)

1<=N<=10^7
分析:
我做这道题是看了这篇博客,http://blog.csdn.net/acdreamers/article/details/8542292 然后觉得说的很对啊,就开始做,然后就敲完了,交上去WA了,然后就开始查看自己哪里错了,中间交了n次,各种WA。。。搞得自己都快崩溃了,于是就把上面那个博客贴的源码交上去,结果WA了,wocao,kengdiea。于是就搜题解,看到下面这个题解就很明白了。参考:http://hzwer.com/3466.html

题解:
求1<=x,y<=N且Gcd(x,y)为素数的数对(x,y)有多少对

枚举每个素数,然后每个素数p对于答案的贡献就是(1 ~ n / p) 中有序互质对的个数
而求1~m中有序互质对x,y的个数,可以令y >= x, 当y = x时,有且只有y = x = 1互质,当y > x时,确定y以后符合条件的个数x就是phiy
所以有序互质对的个数为(1 ~ n/p)的欧拉函数之和乘2减1(要求的是有序互质对,乘2以后减去(1, 1)多算的一次)
那么就只需要先筛出欧拉函数再求个前缀和就可以了

我自己做的就是先求素数,再求欧拉函数,最后求和:

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<algorithm>
using namespace std;
const int N=1e7+7;
typedef long long ll;
int n;
bool vis[N];
ll p[N],phi[N],f[N];
int k=0;
void isprime()
{
    memset(vis,0,sizeof(vis));
    int m=sqrt(n+0.5);
    for(int i=2;i<=n;i++)if(!vis[i]){
        p[k++]=i;
        for(int j=i+i;j<=n;j+=i)vis[j]=1;
    }
}
void phi_table()
{
    memset(phi,0,sizeof(phi));
    phi[1]=1;
    for(int i=2;i<=n;i++)if(!phi[i]){
        for(int j=i;j<=n;j+=i){
            if(!phi[j])phi[j]=j;
            phi[j]=phi[j]/i*(i-1);
        }
    }
}
ll solve()
{
    f[0]=0;
    for(int i=1;i<=n;i++)
        f[i]=f[i-1]+(phi[i]);
    ll ans=0;
    for(int i=0;i<k&&p[i]<=n;i++)
        ans+=f[n/p[i]]*2-1;
    return ans;
}
int main()
{
    scanf("%d",&n);
    isprime();
    phi_table();
    cout<<solve()<<endl;
    return 0;
}

其实更快的算法是线性筛法求解积性函数,也就是上面那个博客上的代码:

#include<iostream>
#include<cstdio>
#define ll long long
#define N 10000005
using namespace std;
int n,p,tot;
int phi[N],pri[1000005];
bool mark[N];
ll ans,sum[N];
void getphi()
{
    phi[1]=1;
    for(int i=2;i<=n;i++)
    {
        if(!mark[i]){phi[i]=i-1;pri[++tot]=i;}
        for(int j=1;j<=tot;j++)
        {
            int x=pri[j];
            if(i*x>n)break;
            mark[i*x]=1;
            if(i%x==0){phi[i*x]=phi[i]*x;break;}
            else phi[i*x]=phi[i]*phi[x];
        }
    }
}
int main()
{
    scanf("%d",&n);
    getphi();
    for(int i=1;i<=n;i++)
        sum[i]=sum[i-1]+phi[i];
    for(int i=1;i<=tot;i++)
        ans+=sum[n/pri[i]]*2-1;
    printf("%lld",ans);
    return 0;
}

做法还有莫比乌斯函数,可是已经凌晨了,还是明天再看看吧,哎。。这题花了我一晚上啊QAQ

posted @ 2016-04-04 00:24  HARD_UNDERSTAND  阅读(625)  评论(0编辑  收藏  举报