zoukankan      html  css  js  c++  java
  • BZOJ 2820: YY的GCD 莫比乌斯反演 + 数学推导 + 线性筛

    求 $sum_{i=1}^{n}sum_{j=1}^{m}[gcd(i,j)in prime]$
     
    按套路前提 $gcd(i,j)$
     
    $Rightarrowsum_{d=1}^{n}din primesum_{i=1}^{n}sum_{j=1}^{m}[gcd(i,j)==d]$
     
    后面的 $sum_{i=1}^{n}sum_{j=1}^{m}[gcd(i,j)==d]$ 是反演模板
     
    $Rightarrow sum_{b=1}^{left lfloor frac{n}{d} ight floor}mu(b)left lfloor frac{n}{db} ight floor left lfloor frac{m}{db} ight floor$
     
    答案为 $sum_{d=1}^{n}din prime sum_{b=1}^{left lfloor frac{n}{d} ight floor}mu(b)left lfloor frac{n}{db} ight floor left lfloor frac{m}{db} ight floor$
     
    用这个式子整除分块算是 $O(n)$ 的,我们优化一下.
     
    令 $T=db$
     
    则原式 $=sum_{T=1}^{n}left lfloor frac{n}{T} ight floorleft lfloor frac{m}{T} ight floorsum_{d|T,din prime}mu(frac{T}{d})$
     
    发现后面的 $sum_{d|T,din prime}mu(frac{T}{d})$ 可以提前预处理(只需枚举质数并暴力更新就行)
    #include <cstdio>
    #include <algorithm>
    #include <cstring>
    #include <vector> 
    #define maxn 10000009
    const long long N = 10000009 ; 
    #define ll long long 
    #define setIO(s) freopen(s".in","r",stdin),freopen(s".out","w",stdout)  
    using namespace std; 
    int vis[maxn],prime[maxn],mu[maxn],g[maxn],tot; 
    long long sumv[maxn]; 
    void init(){
        mu[1]=1; 
        for(int i=2;i<maxn;++i) {
            if(!vis[i]) { prime[++tot]=i,mu[i]=-1; }
            for(int j=1;j<=tot && (ll)prime[j]*i < (ll)maxn;++j) 
            {
                vis[prime[j]*i]=1; 
                if(i % prime[j]==0) {
                    mu[i * prime[j]]=0; 
                    break; 
                }
                mu[i * prime[j]]=-mu[i]; 
            }
        } 
        for(int i=1;i<=tot;++i)
            for(ll j=1;(ll)j*prime[i]<N;++j) 
                g[prime[i]*j]+=mu[j];
        for(int i=1;i<=10000000;++i) sumv[i] = (long long)sumv[i-1]+g[i]; 
    }
    ll work(int n,int m) {
        if(n>m) swap(n,m); 
        long long ans=0; 
        for(ll i=1,j;i<=n;i=j+1) {
            j = min(n/(n/i),m/(m/i)); 
            ans += (n/i) * (m/i) * (sumv[j] - sumv[i-1]); 
        }
        return ans; 
    }
    int main(){
        //setIO("input");
        init();  
        int T,x,y; scanf("%d",&T);
        while(T--) scanf("%d%d",&x,&y),printf("%lld
    ",work(x,y));  
        return 0; 
    }
    

      

  • 相关阅读:
    c++ 中pair类模板的用法详解
    求解Catalan数,(大数相乘,大数相除,大数相加)
    POJ--2823--Sliding Window----单调队列问题
    POJ2796 Feel Good -- 单调队列
    Graham扫描法 --求凸包
    山东理工大学第七届ACM校赛-G 飞花的传送门
    NKOJ1236 a^b (数论定理的应用)
    大数相减 C语言
    SPFA ----模板 O(kE) (k一般不超过2)
    C++大数相加
  • 原文地址:https://www.cnblogs.com/guangheli/p/10439342.html
Copyright © 2011-2022 走看看