zoukankan      html  css  js  c++  java
  • [bzoj2693] jzptab

    Description

    img

    Input

    一个正整数T表示数据组数

    接下来T行 每行两个正整数 表示N、M

    Output

    T行 每行一个整数 表示第i组数据的结果

    Sample Input

    1
    4 5
    

    Sample Output

    122
    

    HINT

    T <= 10000
    N, M<=10000000

    Solution

    前置知识:莫比乌斯反演

    推一下题目中的式子,莫比乌斯反演下,可得:

    [egin{align} ans&=sum_{i=1}^nsum_{j=1}^mfrac{i*j}{gcd(i,j)}\ &=sum_{d=1}^{min(n,m)}dsum_{i=1}^{lfloorfrac{n}{d} floor}sum_{j=1}^{lfloorfrac{m}{d} floor}i*j[gcd(i,j)=1]\ &=sum_{d=1}^{min(n,m)}dsum_{i=1}^{lfloorfrac{n}{d} floor}sum_{j=1}^{lfloorfrac{m}{d} floor}i*j sum_{d^prime|i&d^prime|j}mu(d^prime) \ &=sum_{d=1}^{min(n,m)}dsum_{d^prime}mu(d^prime)f(lfloorfrac{n}{dd^prime} floor,lfloorfrac{m}{dd^prime} floor){d^prime}^2 \ &=sum_{T=1}^{min(n,m)}f(lfloorfrac{n}{T} floor,lfloorfrac{m}{T} floor)sum _{d|T}dmu(frac{T}{d})(frac{T}{d})^2\ &=sum_{T=1}^{min(n,m)}f(lfloorfrac{n}{T} floor,lfloorfrac{m}{T} floor)Tsum _{d|T}mu(d)*d\ end {align} ]

    其中,定义(f)为:

    [egin{align} f(n,m)&=sum_{i=1}^nsum_{j=1}^mi*j\ &=sum_{i=1}^nfrac{i*m*(m+1)}{2}\ &=frac{n*(n+1)*m*(m+1)}{4} end{align} ]

    (f)那部分数论分块搞搞就行了。

    然后考虑(ans)的最后一部分,设为(g),即:

    [g(n)=n*sum_{d|n}mu(d)*d ]

    由于(n)较大,考虑线筛这个东西,设质数(p),当(p mid n)时,展开(g(n*p))

    [egin{align} g(n*p)&=n*p*(sum_{d|n}mu(d)*d+sum_{d|n}mu(d*p)*d*p)\ &=p*(g(n)+g(n)*(-p))\ &=p*(1-p)*g(n)\ &=g(p)*g(n) end{align} ]

    否则同理,可得(g(n*p)=g(n)*p)

    然后,很重要的一点是,1e8+9不是质数!!中间要用到的(inv4)可以线筛或(exgcd)等等

    线筛逆元:

    [inv[i]=(mod-mod/i)*inv[mod\%i]\%mod; ]

    时间复杂度(O(n+qsqrt{n}))

    #include<bits/stdc++.h>
    using namespace std;
    
    #define int long long 
    
    void read(int &x) {
    	x=0;int f=1;char ch=getchar();
    	for(;!isdigit(ch);ch=getchar()) if(ch=='-') f=-f;
    	for(;isdigit(ch);ch=getchar()) x=x*10+ch-'0';x*=f;
    }
    
    void print(int x) {
    	if(x<0) putchar('-'),x=-x;
    	if(!x) return ;print(x/10),putchar(x%10+48);
    }
    void write(int x) {if(!x) putchar('0');else print(x);putchar('
    ');}
    
    const int maxn = 1e7+1;
    const int mod = 100000009;
    
    int pri[maxn],vis[maxn],f[maxn],tot,n,m,k,p[maxn],inv4,inv[maxn];
    
    int qpow(int a,int x) {
        int res=1;
        for(;x;x>>=1,a=a*a%mod) if(x&1) res=res*a%mod;
        return res;
    }
    
    void sieve() {
    	f[1]=1;
    	for(int i=2;i<maxn;i++) {
    		if(!vis[i]) pri[++tot]=i,f[i]=i*(1-i)%mod;
    		for(int j=1;j<=tot&&i*pri[j]<maxn;j++) {
    			vis[i*pri[j]]=1;
    			if(!(i%pri[j])) {f[i*pri[j]]=f[i]*pri[j]%mod;break;}
    			else f[i*pri[j]]=f[i]*f[pri[j]]%mod;
    		}
    	}inv[1]=1;
    	for(int i=1;i<maxn;i++) {
    		if(i!=1) inv[i]=(mod-mod/i)*inv[mod%i]%mod;
    		f[i]=(f[i-1]+f[i])%mod;
    	}
    }
    
    int calc(int n,int m) {return n*(n+1)%mod*m%mod*(m+1)%mod*inv4%mod;}
    
    signed main() {
    	sieve();int t;read(t);inv4=inv[4];
    	while(t--) {
    		int n,m;read(n),read(m);
    		int T=1,ans=0;
    		while(T<=n&&T<=m) {
    			int pre=T;T=min(n/(n/T),m/(m/T));
    			ans=(ans+calc(n/T,m/T)*(f[T]-f[pre-1])%mod)%mod;
    			T++;
    		}
    		write((ans%mod+mod)%mod);
    	}
    	return 0;
    }
    
    
  • 相关阅读:
    【BZOJ 3090】 树形DP
    【BZOJ 2323】 2323: [ZJOI2011]细胞 (DP+矩阵乘法+快速幂*)
    【BZOJ 1019】 1019: [SHOI2008]汉诺塔 (DP?)
    【BZOJ 3294】 3294: [Cqoi2011]放棋子 (DP+组合数学+容斥原理)
    【BZOJ 3566】 3566: [SHOI2014]概率充电器 (概率树形DP)
    【BZOJ 2121】 (字符串DP,区间DP)
    【BZOJ 4305】 4305: 数列的GCD (数论)
    【UOJ 179】 #179. 线性规划 (单纯形法)
    【BZOJ 4568】 4568: [Scoi2016]幸运数字 (线性基+树链剖分+线段树)
    【BZOJ 4027】 4027: [HEOI2015]兔子与樱花 (贪心)
  • 原文地址:https://www.cnblogs.com/hbyer/p/10055115.html
Copyright © 2011-2022 走看看