zoukankan      html  css  js  c++  java
  • [SDOI2015][bzoj 3994][Luogu P3327] 约数个数和 (莫比乌斯反演)

    题目描述

    d(x)d(x)xx的约数个数,给定NNMM,求 i=1Nj=1Md(ij)sum^{N}_{i=1}sum^{M}_{j=1} d(ij)
    N,M,T<=50000N,M,T<=50000

    题目分析

    首先很不显然的有这样一个结论:
    d(ij)=xiyj[(x,y)==1]d(ij)=sum_{x|i}sum_{y|j}[(x,y)==1]

    证明如下

    将一个数唯一质因数分解为p1k1p2k2p3k3...pnknp_1^{k_1}p_2^{k_2}p_3^{k_3}...p_n^{k_n},其约数个数为(k1+1)(k2+1)(k3+1)...(kn+1)(k_1+1)(k_2+1)(k_3+1)...(k_n+1)

    考虑一个质数ppd(ij)d(ij)的影响,假设ii分解质因数后有kkpp,jj分解质因数后有qqpp,则产生的贡献即为k+q+1k+q+1,接下来是关键的一步(反正我想不到XD)
    k+q+1=x=0ky=0q[(px,py)==1]k+q+1=sum_{x=0}^ksum_{y=0}^q[(p^x,p^y)==1]

    (此句可自行忽略:只有当xy=0xy=0时才会有值,这就相当于一个(k+1)(q+1)(k+1)∗(q+1)的矩形,只取了左下角的LL字型的一块)

    根据乘法原理,有
    d(ij)=x1=0k1y1=0q1[(p1x1,p1y1)==1]x2=0k2y2=0q2[(p2x2,p2y2)==1]xn=0knyn=0qn[(pnxn,pnyn)==1] d(ij)=sum_{x_1=0}^{k_1}sum_{y_1=0}^{q_1}[(p_1^{x_1},p_1^{y_1})==1] ewline *sum_{x_2=0}^{k_2}sum_{y_2=0}^{q_2}[(p_2^{x_2},p_2^{y_2})==1] ewline ··· ewline *sum_{x_n=0}^{k_n}sum_{y_n=0}^{q_n}[(p_n^{x_n},p_n^{y_n})==1]
    因为必须满足所有(pnxn,pnyn)==1(p_n^{x_n},p_n^{y_n})==1才会对答案造成贡献,则可以变形为(i=1npnxn,i=1npnyn)==1(prod_{i=1}^np_n^{x_n},prod_{i=1}^np_n^{y_n})==1

    所以d(ij)=xiyj[(x,y)==1]d(ij)=sum_{x|i}sum_{y|j}[(x,y)==1]

    (以上证明摘自:传送门)

    证毕

    现在就有了Ans=i=1Nj=1Mxiyj[(x,y)==1]Ans=sum_{i=1}^{N}sum_{j=1}^{M}sum_{x|i}^{}sum_{y|j}^{}[(x,y)==1]

    根据数论中切换枚举次序的套路以及莫比乌斯反演对布尔条件框的替换,我们得到
    Ans=x=1Ny=1MNxMy[(x,y)==1]=x=1Ny=1MNxMyk(x,y)μ(k)=k=1min(N,M)μ(k)kxNxkyMy=k=1min(N,M)μ(k)x=1NkNkxy=1MkMky Ans=sum_{x=1}^{N}sum_{y=1}^{M}⌊frac{N}{x}⌋⌊frac{M}{y}⌋[(x,y)==1] ewline =sum_{x=1}^{N}sum_{y=1}^{M}⌊frac{N}{x}⌋⌊frac{M}{y}⌋sum_{k|(x,y)}mu(k) ewline =sum_{k=1}^{min(N,M)}mu(k)sum_{k|x}^{}⌊frac{N}{x}⌋sum_{k|y}^{}⌊frac{M}{y}⌋ ewline =sum_{k=1}^{min(N,M)}mu(k)sum_{x=1}^{⌊frac{N}{k}⌋}⌊frac{N}{kx}⌋sum_{y=1}^{⌊frac{M}{k}⌋}⌊frac{M}{ky}⌋

    于是这里使用分块优化,预处理μmu的前缀和

    最后考虑后面的两个Sigma怎么处理
    可以观察发现它们都是i=1nnisum_{i=1}^{n}⌊frac{n}{i}⌋的形式,此处可以用分块优化Θ(nn)Theta(nsqrt{n})预处理

    其实还可以Θ(n)Theta(n)预处理,可以发现i=1nni=i=1nidn1sum_{i=1}^{n}⌊frac{n}{i}⌋=sum_{i=1}^{n}sum_{i|d}^{n}1
    显然是约数个数和d(n)d(n)的前缀和函数,于是线性筛出μ(n)mu(n)d(n)d(n),求出前缀和即可

    每次询问Θ(n)Theta(sqrt{n}), 预处理Θ(n)Theta(n),总时间复杂度为Θ(nn)Theta(nsqrt{n})

    TIPS

    线性筛约数个数时存一下最小的质数p1p_1的次方+1+1,即存下(k1+1)(k_1+1)就可以方便的线性筛了

    线性筛约数和同理,存一下最小的质数p1p_1所对应的首项为11,公比为p1p_1,项数为k1+1k_1+1的等比数列,即存下i=1k1p1isum_{i=1}^{k_1}p_1^i就可以愉快的线性筛了

    AC code
    #include <cstdio>
    #include <cstring>
    #include <algorithm>
    using namespace std;
    
    const int MAXN = 100001;
    namespace Mobius
    {
    	#define LL long long
    	int Prime[MAXN], Cnt, mu[MAXN], d[MAXN], Minseq[MAXN];
    	bool IsnotPrime[MAXN];
    	LL sum_MU[MAXN], sum_D[MAXN];
    	void init()
    	{
    		mu[1] = d[1] = Minseq[1] = 1;
    		for(int i = 2; i < MAXN; ++i)
    		{
    			if(!IsnotPrime[i])
    				Prime[++Cnt] = i, mu[i] = -1, d[i] = Minseq[i] = 2;
    			for(int j = 1; j <= Cnt && i * Prime[j] < MAXN; ++j)
    			{
    				IsnotPrime[i * Prime[j]] = 1;
    				if(i % Prime[j] == 0)
    				{
    					Minseq[i * Prime[j]] = Minseq[i]+1;
    					mu[i * Prime[j]] = 0;
    					d[i * Prime[j]] = d[i] / Minseq[i] * Minseq[i * Prime[j]];
    					break;
    				}
    				Minseq[i * Prime[j]] = 2;
    				mu[i * Prime[j]] = -mu[i];
    				d[i * Prime[j]] = d[i]<<1;
    			}
    		}
    		for(int i = 1; i < MAXN; ++i)
    			sum_MU[i] = sum_MU[i-1] + mu[i],
    			sum_D[i] = sum_D[i-1] + d[i];
    	}
    	LL calc(int N, int M)
    	{
    		if(N > M) swap(N, M);
    		LL ret = 0;
    		for(int i = 1, j; i <= N; i=j+1)
    		{
    			j = min(N/(N/i), M/(M/i));
    			ret += (sum_MU[j]-sum_MU[i-1]) * sum_D[N/i] * sum_D[M/i];
    		}
    		return ret;
    	}
    }
    using namespace Mobius;
    int main ()
    {
    	init();
    	int T, n, m;
    	scanf("%d", &T);
    	while(T--)
    	{
    		scanf("%d%d", &n, &m);
    		printf("%lld
    ", calc(n, m));
    	}
    }
    
  • 相关阅读:
    Flask把变量注册到模板中
    $.each与$(data).each区别
    【Python备忘】python判断文件和文件夹是否存在
    ISP图像质量自动化测试方法
    使用微软的(how-old.net)构建智能门店管理系统
    在virtualenv中安装libxml2和libxslt
    Python 装饰器学习以及实际使用场景实践
    tensorflow零起点快速入门(4) --入门常用API
    tensorflow零起点快速入门(3)
    树莓派和STM32通过USB和串口通信记录
  • 原文地址:https://www.cnblogs.com/Orz-IE/p/12039472.html
Copyright © 2011-2022 走看看