码迷,mamicode.com
首页 > 其他好文 > 详细

BZOJ 3994 Sdoi2015 约数个数和 莫比乌斯反演

时间:2015-04-16 17:33:50      阅读:148      评论:0      收藏:0      [点我收藏+]

标签:bzoj   bzoj3994   莫比乌斯反演   数论   

题目大意:求ni=1mj=1d(ij)
首先我们有一个很神的结论:
ni=1mj=1d(ij)=ni=1mj=1?ni??mj?[gcd(i,j)==1]
这个结论是怎么来的呢?我们可以先证明这个:
d(nm)=i|nj|m1?1[gcd(i,j)==1]
显然这个式子的前缀和就是上面的式子
现在我们来证明这个式子是对的
我们分开讨论每一个质数p对答案的贡献
不妨设n=n?pk1,m=m?pk2
那么左式中p的贡献显然是k1+k2+1
右式中只考虑p的话,满足要求的数对(i,j)只有(pk1,1),(pk1?1,1),...,(p,1),(1,1),(1,p),...,(1,pk2?1),(1,pk2),共有k1+k2+1
因此等式成立,原式得证。
然后怎么搞还用我说么= =?

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
#define M 50500
using namespace std;

int n,m;

int mu[M],d[M],cnt[M],prime[M],tot;
bool not_prime[M];
void Linear_Shaker()
{
    int i,j;
    mu[1]=1;d[1]=1;
    for(i=2;i<=50000;i++)
    {
        if(!not_prime[i])
        {
            prime[++tot]=i;
            mu[i]=-1;
            d[i]=2;
            cnt[i]=1;
        }
        for(j=1;prime[j]*i<=50000;j++)
        {
            not_prime[prime[j]*i]=true;
            if(i%prime[j]==0)
            {
                mu[prime[j]*i]=0;
                d[prime[j]*i]=d[i]/(cnt[i]+1)*(cnt[i]+2);
                cnt[prime[j]*i]=cnt[i]+1;
                break;
            }
            mu[prime[j]*i]=-mu[i];
            d[prime[j]*i]=d[i]<<1;
            cnt[prime[j]*i]=1;
        }
    }
    for(i=1;i<=50000;i++)
    {
        mu[i]+=mu[i-1];
        d[i]+=d[i-1];
    }
}
long long Solve()
{
    long long re=0;
    int i,last;
    if(n>m) swap(n,m);
    for(i=1;i<=n;i=last+1)
    {
        last=min(n/(n/i),m/(m/i));
        re+=(long long)(mu[last]-mu[i-1])*d[n/i]*d[m/i];
    }
    return re;
}
int main()
{
    int T;
    Linear_Shaker();
    for(cin>>T;T;T--)
    {
        scanf("%d%d",&n,&m);
        printf("%lld\n",Solve());
    }
    return 0;
}

BZOJ 3994 Sdoi2015 约数个数和 莫比乌斯反演

标签:bzoj   bzoj3994   莫比乌斯反演   数论   

原文地址:http://blog.csdn.net/popoqqq/article/details/45078079

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!