zoukankan      html  css  js  c++  java
  • BZOJ 2818

    2818:GCD

    Description

    给定整数$N$,求$1le x,yle N$且$gcd{x,y}$为素数的数对$(x,y)$有多少对.

    Input

    $N$

    Output

    RT

    Sample

    Input
    ----------------
    4
    
    Output
    ----------------
    4

    Hint

    4个解为$(2,2)$,$(3,3)$,$(2,4)$,$(4,2)$

    $1le Nle 10^7$

    Source

    湖北省队互测

    这道题目很有意思.

    我们将一个数$A$质因数分解:$A=p_1^{m_1}p_2^{m_2}dots p_n^{m_n}$其中$p_i$是连续素数,记为$FS(A)=left[m_1,m_2dots ight]$.对于每一个数来说他们的分解都是唯一的(算术基本定理).

    我们将$FS(A)=a$的反函数记为$SF(a)=A$.

    那么$gcd{A,B}=SF([(min{FS(A)[0],FS(B)[0]}),dots])$.

    那么,易证$gcd{frac{A}{m},frac{B}{m}}=frac{gcd{a,b}}{m}$.

    因此,我们可以将原问题稍稍转化下,我们可以求$1le x,yle lfloorfrac{N}{p} floor$(其中$p$是一个素数)时$gcd{x,y}=1$的组数.注意到这个问题中$x,y$互质,那么我们从下到大枚举$y$,算出$phi(y)$的和相加.由于考虑顺序,组数要乘以$2$,由于$(1,1)$正反是一样的,那么再减去$1$.

    如何求$phi(y)$呢?我们可以用线性筛法稍微改动一下.具体的看我程序.

    剩下的也不难想到了.我们前缀和了$phi(y)$,那么对于每个素数的计算就变成$O(1)$了.

    #include <cstdio>
    int N,i,j,k,phi[10000001],primes[700000],pr;
    long long phipref[10000001],res;
    int np[10000001];
    void genPhi(int n){
    	phipref[1]=phi[1]=1;
    	for(i=2;i<=n;++i){
    		if(!np[i]){
    			phi[i]=i-1;
    			np[i]=primes[pr++]=i;
    		}
    		for(j=0;j<pr && (k=primes[j]*i)<=n;++j){
    			np[k]=primes[j];
    			if(np[i]==primes[j]){
    				phi[k]=phi[i]*primes[j];
    				break;
    			}else{
    				phi[k]=phi[i]*(primes[j]-1);
    			}
    		}
    		phipref[i]=phipref[i-1]+phi[i];
    	}
    }
    int main(){
    	scanf("%d",&N);
    	genPhi(N);
    	for(i=0;i<pr;++i) res+=phipref[N/primes[i]]*2-1;
    	printf("%lld
    ",res);
    	return 0;
    }
    

    * 类似题目:BZOJ 2190 < 这道题我不清楚原做法是怎么样的0.0,数据范围开小了太多.

  • 相关阅读:
    故乡
    webService和Restful
    java多线程(六)线程控制类
    java内存模型
    java多线程(五)线程通讯
    java多线程(四)死锁
    java多线程(三)线程的安全问题
    java多线程(二)线程的生命周期
    java多线程(一)创建线程的四种方式
    Spring Cloud(一)简单的微服务集成Eureka
  • 原文地址:https://www.cnblogs.com/tmzbot/p/4462740.html
Copyright © 2011-2022 走看看