zoukankan      html  css  js  c++  java
  • luogu P2522 [HAOI2011]Problem b |莫比乌斯反演+容斥原理

    题目描述

    对于给出的 (n) 个询问,每次求有多少个数对 ((x,y)),满足 (a le x le b)(c le y le d),且 (gcd(x,y) = k)(gcd(x,y)) 函数为 (x)(y) 的最大公约数。

    输入格式

    第一行一个整数 (n),接下来 (n) 行每行五个整数,分别表示 (a,b,c,d,k)

    输出格式

    (n) 行,每行一个整数表示满足要求的数对 ((x,y)) 的个数。


    (f(x)=sum_{x=1}^asum_{y=1}^b [gcd(x,y)=d])

    容斥一下减去不是要求的区间

    #include<cmath>
    #include<vector>
    #include<cstdio>
    #include<cstring>
    #include<iostream>
    #include<algorithm>
    using namespace std;
    const int N=1e7+5;
    #define ll long long
    int vis[N],mu[N],tot,n;
    int pri[N];
    int Mobius(){
    	vis[1]=mu[1]=1;
    	for(int i=2;i<=n;i++){
    		if(!vis[i])pri[++tot]=i,mu[i]=-1;
    		for(int j=1;j<=tot&&i*pri[j]<=n;j++){
    			vis[i*pri[j]]=1;
    			if(i%pri[j])mu[i*pri[j]]=-mu[i];
    			else { mu[i*pri[j]]=0; break; }
    		}
    	}
    	for(int i=1;i<=n;i++)
    	mu[i]+=mu[i-1],vis[i]=(!vis[i])+vis[i-1];
    }
    inline ll calc(int a){
    	ll ans=0;
    	for(int i=1,j=0;i<=a;i=j+1){
    		j=a/(a/i);
    		ans+=1ll*(mu[j]-mu[i-1])*(a/i)*(a/i);
    	}
    	return ans;
    }
    signed main(){
    	scanf("%d",&n);
    	Mobius();
    	ll ans=0;
    	for(int i=1,j=0;i<=n;i=j+1){
    		j=n/(n/i);
    		ans+=1ll*(vis[j]-vis[i-1])*calc(n/i);
    	}
    	cout<<ans<<endl;
    }
    
  • 相关阅读:
    JAVA类与对象(一)----基础概念理解
    JAVA数组
    JAVA算术运算符、关系运算符和位运算符
    JAVA数据类型
    JAVA配置环境变量
    C#中怎么生成36位的GUID
    jqAjax
    二进制1的个数
    成绩转换
    对决
  • 原文地址:https://www.cnblogs.com/naruto-mzx/p/12973881.html
Copyright © 2011-2022 走看看