洛谷P2522 - [HAOI2011]Problem b

Portal

Description

进行\(T(T\leq10^5)\)次询问,每次给出\(x_1,x_2,y_1,y_2\)和\(d\)(均不超过\(10^5\)),求\(\sum_{i=x_1}^{x_2} \sum_{j=y_1}^{y_2} [gcd(i,j)=d]\)。

Solution

莫比乌斯反演入门题。
设\(calc(n,m)\)表示\(i\in[1,n],j\in[1,m]\)且\(gcd(i,j)=d\)的数对\((i,j)\)的个数。那么简单地进行容斥,可知\(ans=calc(x_2,y_2)-calc(x_1-1,y_2)-calc(x_2,y_1-1)+calc(x_1-1,x_2-1)\)。
于是考虑如何计算\(calc(n,m)\)。
\[ f(d) = \sum_{i=1}^n \sum_{j=1}^m [gcd(i,j)=d] \]

\[\begin{align*}
F(x) &= \sum_{x|d} f(d) \ &= \sum_{x|d} \sum_{i=1}^n \sum_{j=1}^m [gcd(i,j)=d] \ &= \sum_{k=1}^{?\frac{n}{x}?} \sum_{i=1}^n \sum_{j=1}^m [gcd(i,j)=kx] \ &= ?\frac{n}{x}??\frac{m}{x}?
\end{align*}\] \(gcd(i,j)=kx \Leftrightarrow x|i\)且\(x|j\),那满足条件的\((i,j)\)就有\(?\frac{n}{x}??\frac{m}{x}?\)对。再进行莫比乌斯反演:
\[ f(x)= \sum_{x|d} \mu(\frac{d}{x}) F(d) = \sum_{x|d} \mu(\frac{d}{x})?\frac{n}{d}??\frac{m}{d}? = \sum_{k=1}^{?\frac{n}{x}?} \mu(k)?\frac{n}{kx}??\frac{m}{kx}? \]这个做法看起来是\(O(\dfrac{n}{x})\)的。不过由于\(?\dfrac{n}{i}?\)最多只有\(\sqrt n\)种取值,所以我们可以以\(O(\sqrt n)\)的复杂度进行计算。
|i|1|2|3|4|5|6|7|8|9|10|11|12|13|14|15|
|-|-|-|-|-|-|-|-|-|-|--|--|--|--|--|--|
|n|15|7|5|3|3|2|2|1|1|1|1|1|1|1|1|

abc def hijklkjlk
sldkf lskdfkd lskdjflksd

观察发现,一个取值为\(v\)的区间是以\(?\frac{n}{v}?\)结尾的,下一个区间是从\(?\frac{n}{v}?+1\)开始的,模拟这一性质去计算即可。若对于区间\(k\in[L,R]\)有\(?\dfrac{n}{kx}?=v_1,?\dfrac{m}{kx}?=v_2\),那么该区间对答案的贡献为\(v_1v_2\sum_{k=L}^R \mu(k)\),预处理出\(\mu(x)\)的前缀和即可。

时间复杂度\(O(T\sqrt {10^5})\)。

Code

//[HAOI2011]Problem b
#include <algorithm>
#include <cstdio>
using std::min; using std::swap;
typedef long long lint;
inline char gc()
{
    static char now[1<<16],*s,*t;
    if(s==t) {t=(s=now)+fread(now,1,1<<16,stdin); if(s==t) return EOF;}
    return *s++;
}
inline int read()
{
    int x=0; char ch=gc();
    while(ch<'0'||'9'<ch) ch=gc();
    while('0'<=ch&&ch<='9') x=x*10+ch-'0',ch=gc();
    return x;
}
const int N=5e4+10;
int mu[N],pre[N];
int cntP,pr[N]; bool notP[N];
void getMu(int n)
{
    mu[1]=1;
    for(int i=2;i<=n;i++)
    {
        if(!notP[i]) pr[++cntP]=i,mu[i]=-1;
        for(int j=1;j<=cntP;j++)
        {
            if((lint)i*pr[j]>n) break;
            int x=i*pr[j]; notP[x]=true;
            if(i%pr[j]) mu[x]=-mu[i]; else {mu[x]=0; break;}
        }
    }
    for(int i=1;i<=n;i++) pre[i]=pre[i-1]+mu[i];
}
int k;
lint calc(int x,int y)
{
    x/=k,y/=k; if(x>y) swap(x,y);
    lint res=0;
    for(int L=1,R;L<=x;L=R+1)
    {
        int v1=x/L,v2=y/L; R=min(x/v1,y/v2);
        res+=1LL*(pre[R]-pre[L-1])*v1*v2;
    }
    return res;
}
int main()
{
    getMu(5e4);
    int Q=read();
    while(Q--)
    {
        int fr1=read(),to1=read(),fr2=read(),to2=read(); k=read();
        printf("%lld\n",calc(to1,to2)-calc(fr1-1,to2)-calc(to1,fr2-1)+calc(fr1-1,fr2-1));
    }
    return 0;
}

原文地址:https://www.cnblogs.com/VisJiao/p/LgP2522.html

时间: 2024-10-08 12:47:36

洛谷P2522 - [HAOI2011]Problem b的相关文章

P2522 [HAOI2011]Problem b

题目描述 对于给出的n个询问,每次求有多少个数对(x,y),满足a≤x≤b,c≤y≤d,且gcd(x,y) = k,gcd(x,y)函数为x和y的最大公约数. 输入输出格式 输入格式: 第一行一个整数n,接下来n行每行五个整数,分别表示a.b.c.d.k 输出格式: 共n行,每行一个整数表示满足要求的数对(x,y)的个数 输入输出样例 输入样例#1: 2 2 5 1 5 1 1 5 1 5 2 输出样例#1: 14 3 说明 100%的数据满足:1≤n≤50000,1≤a≤b≤50000,1≤c

Luogu P2522 [HAOI2011]Problem b 莫比乌斯反演

设$f(d)=\sum_{i=1}^N\sum_{j=1}^M[gcd(i,j)==d],\\F(n)=\sum_{n|d}f(d)=\lfloor \frac{N}{n} \rfloor \lfloor \frac{M}{n} \rfloor$ 则$f(n)$ $=\sum_{n|d}\mu(\frac{n}{d})F(d)$ $=\sum_{n|d}\mu(\frac{n}{d})\lfloor \frac{N}{d} \rfloor \lfloor \frac{M}{d} \rfloor$

P2522 [HAOI2011]Problem b 题解

莫比乌斯反演 ACWing215的升级版 直接计算啊a<=i<=b,c<=j<=d的gcd(x,y)==k的个数不太好计算,因为我们很多时候都是从一开始枚举,而不是从一个大于1的数开始枚举,[x,y]范围内d的倍数也无法直接计算 仔细一看这就是一个二维偏序,二维前缀和即可 #include<bits/stdc++.h> using namespace std; #define go(i,a,b) for(int i=a;i<=b;++i) typedef long

洛谷1001 A+B Problem

洛谷1001 A+B Problem 本题地址:http://www.luogu.org/problem/show?pid=1001 题目描述 输入两个整数a,b,输出它们的和(|a|,|b|<=10^9).注意1.pascal使用integer会爆掉哦!2.有负数哦!3.c/c++的main函数必须是int类型,而且最后要return 0.这不仅对洛谷其他题目有效,而且也是noip/noi比赛的要求! 好吧,同志们,我们就从这一题开始,向着大牛的路进发.“任何一个伟大的思想,都有一个微不足道的

洛谷 P1001 A+B Problem

题目描述 输入两个整数a,b,输出它们的和(|a|,|b|<=10^9). 注意 1.pascal使用integer会爆掉哦! 2.有负数哦! 3.c/c++的main函数必须是int类型,而且最后要return 0.这不仅对洛谷其他题目有效,而且也是noip/noi比赛的要求! 好吧,同志们,我们就从这一题开始,向着大牛的路进发. “任何一个伟大的思想,都有一个微不足道的开始.” 输入输出格式 输入格式: 两个整数以空格分开 输出格式: 一个数 输入输出样例 输入样例#1: 20 30 输出样

洛谷.1919.[模板]A乘B Problem升级版(FFT)

题目链接:洛谷.BZOJ2179 //将乘数拆成 a0*10^n + a1*10^(n-1) + ... + a_n-1的形式 //可以发现多项式乘法就模拟了竖式乘法 所以用FFT即可 注意处理进位 //n位*n位最多就只有2n位了 //论putchar的速度..还是快的 #include <cmath> #include <cstdio> #include <cctype> #include <algorithm> #define gc() getchar

洛谷1231 教辅的组成

洛谷1231 教辅的组成 https://www.luogu.org/problem/show?pid=1231 题目背景 滚粗了的HansBug在收拾旧语文书,然而他发现了什么奇妙的东西. 题目描述 蒟蒻HansBug在一本语文书里面发现了一本答案,然而他却明明记得这书应该还包含一份练习题.然而出现在他眼前的书多得数不胜数,其中有书,有答案,有练习册.已知一个完整的书册均应该包含且仅包含一本书.一本练习册和一份答案,然而现在全都乱做了一团.许多书上面的字迹都已经模糊了,然而HansBug还是可

洛谷 P2801 教主的魔法 题解

此文为博主原创题解,转载时请通知博主,并把原文链接放在正文醒目位置. 题目链接:https://www.luogu.org/problem/show?pid=2801 题目描述 教主最近学会了一种神奇的魔法,能够使人长高.于是他准备演示给XMYZ信息组每个英雄看.于是N个英雄们又一次聚集在了一起,这次他们排成了一列,被编号为1.2.…….N. 每个人的身高一开始都是不超过1000的正整数.教主的魔法每次可以把闭区间[L, R](1≤L≤R≤N)内的英雄的身高全部加上一个整数W.(虽然L=R时并不

洛谷 P3367 并查集模板

#include<cstdio> using namespace std; int n,m,p; int father[2000001]; int find(int x) { if(father[x]!=x) father[x]=find(father[x]); return father[x]; } void unionn(int i,int j) { father[j]=i; } int main() { scanf("%d%d",&n,&m); for