BZOJ 2005 NOI2010 能量采集 莫比乌斯反演

BZOJ又被卡了

这里写图片描述


这道题水过,作为第一道莫比乌斯反演的习题,我还是理解了好长时间分块的想法的,可能是我太菜了吧…


首先一个推导我们需要求的是

i=0nj=0m(gcd(i,j)21)

进行推导
=2i=0nj=0m(gcd(i,j))nm

n=d|nφ(d),ngcd(i,j),

=2i=0nj=0md|gcd(i,j)φ(d)nm

=2i=0nj=0md=1n[d|i][d|j]φ(d)nm

=2d=1ni=0n[d|i]j=0m[d|j]φ(d)nm

=2d=1nn/dm/dφ(d)nm

一组询问可以O(n)水过去,多组询问就分块搞
注意其中有一步,是直接把d转换到1到n了,其实这里有个取较小值操作,我们可以把n当作小的那个,然后来搞


#include<iostream>
#include<cstring>
#include<cstdio>
using namespace std;
const int MAXN=100000+50;
long long n,m,is_not_prime[MAXN],phi[MAXN],prime[MAXN];
inline void getphi(){
    phi[1]=1;
    for(register int i=2;i<=n;i++){
        if(!is_not_prime[i]){
            prime[++prime[0]]=i;
            phi[i]=i-1;
        }
        for(register int j=1;j<=prime[0]&&i*prime[j]<=n;j++){
            is_not_prime[i*prime[j]]=1;
            if(i%prime[j]==0){
                phi[i*prime[j]]=prime[j]*phi[i];
                break;
            }else{
                phi[i*prime[j]]=phi[i]*(prime[j]-1);
            } 
        }
        phi[i]+=phi[i-1];
    }
}
int main(){
    scanf("%lld%lld",&n,&m);
    if(n<m) swap(n,m);
    getphi();
    long long end,final=0;
    for(register long long start=1;start<=m;start=end+1){
        end=min(n/(n/start),m/(m/start));
        final+=(phi[end]-phi[start-1])*(n/start)*(m/start);
    }
    printf("%lld\n",final*2-m*n);
    return 0;
}

这里写图片描述

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值