【CodeForces】870 F. Paths

【题目】F. Paths

【题意】给定数字n,图上有编号为1~n的点,两点当且仅当gcd(u,v)≠1时有连边,定义d(u,v)为两点间最短距离(若不连通则为0),求Σd(u,v),1<=u<v<=n,n<=10^7。

【算法】数论

【题解】对于1<=x<=n,当x=1或x是大于n/2的素数时,x是孤立节点。

令p[x]表示x的最小素因子,考虑任意一对点的连边情况:

1.d=0:当至少一个点是孤立节点时,d(u,v)=0,否则都能通过下面的情况到达。

2.d=1:当gcd(u,v)≠1时,对数为one=Σx-1-φ(x),1<=x<=n。

3.d=2:p[u]*p[v]<=n时,可以通过u - p[u]*p[v] - v这条路径到达。(思路:两边最小素因子的乘积是最小中转点)

4.d=3:通过 u - 2*p[u] - 2*p[v] - v 到达。(思路:用最小的素因子2沟通)

通过情况1,容易计算出除了孤立节点外的点对总数sum。

情况2也容易计算(记为one),那么只须计算出情况3,情况4也可以用sum减去2和3得到。

如何计算满足p[u]*p[v]<=n&&gcd(u,v)=1的点对数?先计算满足p[u]*p[v]<=n(1<=u,v<=n)的点对数,然后减去u=v的情况,除以2后再减去one(u,v互素)即可得到。

复杂度O(n)。

#include<cstdio>
#include<cstring>
#include<algorithm>
#define ll long long
using namespace std;
const int maxn=10000010;
int prime[maxn],p[maxn],n,tot,phi[maxn],num[maxn],can;
ll one=0,two=0,three=0,s[maxn];
int main(){
    scanf("%d",&n);
    phi[1]=1;
    for(int i=2;i<=n;i++){
        if(!p[i]){p[i]=prime[++tot]=i;phi[i]=i-1;}
        for(int j=1;j<=tot&&i*prime[j]<=n;j++){
            p[i*prime[j]]=prime[j];
            if(i%prime[j]==0){phi[i*prime[j]]=phi[i]*prime[j];break;}
            phi[i*prime[j]]=phi[i]*(prime[j]-1);
        }
    }
    for(int i=2;i<=n;i++)one+=i-1-phi[i];
    for(int i=2;i<=n;i++)num[p[i]]++;
    for(int i=2;i<=n;i++)s[i]=s[i-1]+num[i];
    for(int i=2;i<=n;i++)two+=1ll*num[i]*s[n/i];
    for(int i=2;i<=n;i++)if(1ll*p[i]*p[i]<=n)two--;
    two=two/2-one;
    three=0;can=n-1;
    for(int i=1;i<=tot;i++)if(prime[i]*2>n)can--;
    three=1ll*can*(can-1)/2-two-one;
    printf("%lld",one*1+two*2+three*3);
    return 0;
}//learn from 1234567891
View Code

 

posted @ 2018-01-10 16:52  ONION_CYC  阅读(226)  评论(0编辑  收藏  举报