https://www.luogu.org/problemnew/show/P2398

$原式=\sum_{k=1}^n(k\sum_{i=1}^n\sum_{j=1}^n[(i,j)=k])$

方法1:

发现暴力枚举k,就变成这道模板题

复杂度O(nlogn)

 #pragma GCC optimize("Ofast")
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<vector>
using namespace std;
#define fi first
#define se second
#define mp make_pair
#define pb push_back
typedef long long ll;
typedef unsigned long long ull;
typedef pair<int,int> pii;
#define N 2000000
ll prime[N+],len,mu[N+];
bool nprime[N+];
ll n,ans,a2;
ll F(ll x) {return (n/x)*(n/x);}
int main()
{
ll i,j,k;
mu[]=;
for(i=;i<=N;i++)
{
if(!nprime[i]) prime[++len]=i,mu[i]=-;
for(j=;j<=len&&i*prime[j]<=N;j++)
{
nprime[i*prime[j]]=;
if(i%prime[j]==) {mu[i*prime[j]]=;break;}
else mu[i*prime[j]]=-mu[i];
}
}
scanf("%lld",&n);
for(k=;k<=n;k++)
{
a2=;
for(i=;i<=n/k;i++)
a2+=mu[i]*F(i*k);
ans+=a2*k;
}
printf("%lld",ans);
return ;
}

方法2:

https://www.luogu.org/blog/Cinema/solution-p1390

(这是一道类似的题的题解)

复杂度好像是O(n)

具体地说,是嵌套整除分块,第一重枚举n/d,m/d,第二重枚举(n/d)/x,(m/d)/x


方法3:

$\sum_{i=1}^n\sum_{j=1}^n[(i,j)=k]=\sum_{i=1}^{{\lfloor}n/k{\rfloor}}\sum_{j=1}^{{\lfloor}n/k{\rfloor}}[(i,j)=1]$

$=2*(\sum_{i=1}^{{\lfloor}n/k{\rfloor}}\sum_{j=1}^{i}[(i,j)=1])-1=2*(\sum_{i=1}^{{\lfloor}n/k{\rfloor}}\varphi(i))-1$

这个欧拉函数的前缀和可以先预处理出来,然后枚举k

复杂度O(n)

 #include<cstdio>
#include<algorithm>
#include<cstring>
#include<vector>
using namespace std;
#define fi first
#define se second
#define mp make_pair
#define pb push_back
typedef long long ll;
typedef unsigned long long ull;
typedef pair<int,int> pii;
#define N 2000000
ll prime[N+],len,phi[N+];
ll d[N+];
bool nprime[N+];
ll n,ans;
ll F(ll x) {return (n/x)*(n/x);}
int main()
{
ll i,j,k;
phi[]=;
for(i=;i<=N;i++)
{
if(!nprime[i]) prime[++len]=i,phi[i]=i-;
for(j=;j<=len&&i*prime[j]<=N;j++)
{
nprime[i*prime[j]]=;
if(i%prime[j]==) {phi[i*prime[j]]=phi[i]*prime[j];break;}
else phi[i*prime[j]]=phi[i]*(prime[j]-);
}
}
for(i=;i<=N;i++) d[i]+=d[i-]+phi[i];
scanf("%lld",&n);
for(k=;k<=n;k++) ans+=k*(*d[n/k]-);
printf("%lld",ans);
return ;
}

方法4:

注意到方法3第35行的可以用整除分块优化,甚至可以O(n)预处理后每次O(sqrt(n))回答


类似的题(题面并不完全一样)

https://www.luogu.org/problemnew/show/UVA11417

https://www.luogu.org/problemnew/show/UVA11426

https://www.luogu.org/problemnew/show/P1390


https://www.luogu.org/problemnew/show/UVA11424

这题做法稍微有点不一样,因为数据组数多,不能直接每次O(n)回答

可以用方法4水过去

 #include<cstdio>
#include<algorithm>
#include<cstring>
#include<vector>
using namespace std;
#define fi first
#define se second
#define mp make_pair
#define pb push_back
typedef long long ll;
typedef unsigned long long ull;
typedef pair<int,int> pii;
#define N 2000000
ll prime[N+],len,phi[N+];
ll d[N+];
bool nprime[N+];
ll n,ans;
ll F(ll x) {return (n/x)*(n/x);}
int main()
{
ll i,j,k;
phi[]=;
for(i=;i<=N;i++)
{
if(!nprime[i]) prime[++len]=i,phi[i]=i-;
for(j=;j<=len&&i*prime[j]<=N;j++)
{
nprime[i*prime[j]]=;
if(i%prime[j]==) {phi[i*prime[j]]=phi[i]*prime[j];break;}
else phi[i*prime[j]]=phi[i]*(prime[j]-);
}
}
for(i=;i<=N;i++) d[i]+=d[i-]+phi[i];
while(){
scanf("%lld",&n);
if(n==) break;
ans=;
for(i=;i<=n;i=j+)
{
j=min(n,n/(n/i));
ans+=(i+j)*(j-i+)/*(*d[n/i]-);
}
printf("%lld\n",(ans-(n+)*n/)/);}
return ;
}

或者:

设sum[x]=$\sum_{i=1}^{x-1}(i,x)$

那么询问为n时,答案就是$\sum_{i=1}^n{sum[i]}$

而sum[x]=$\sum_{k=1}^{x-1}(k*\sum_{i=1}^{x-1}[(i,x)=k])$

显然只有当k是x的因子且k不等于x时才能有贡献

枚举k,$\sum_{i=1}^{x-1}[(i,x)=k]=\sum_{i=1}^{\frac{x}{k}-1}[(i,\frac{x}{k})=1]$

$=(\sum_{i=1}^{\frac{x}{k}}[(i,\frac{x}{k})=1])=\phi(\frac{x}{k})$

可以看出来sum[x]=$(\sum_{d|x且d{\neq}x}(d*\phi(\frac{x}{d})))$

实际实现时可以把枚举因子变为枚举倍数

复杂度所有数据加起来O(nlogn+q)

 #include<cstdio>
#include<algorithm>
#include<cstring>
#include<vector>
using namespace std;
#define fi first
#define se second
#define mp make_pair
#define pb push_back
typedef long long ll;
typedef unsigned long long ull;
typedef pair<int,int> pii;
#define N 200000
ll prime[N+],len,phi[N+];
ll sum[N+];
bool nprime[N+];
ll n,ans[N+];
int main()
{
ll i,j,k;
phi[]=;
for(i=;i<=N;i++)
{
if(!nprime[i]) prime[++len]=i,phi[i]=i-;
for(j=;j<=len&&i*prime[j]<=N;j++)
{
nprime[i*prime[j]]=;
if(i%prime[j]==) {phi[i*prime[j]]=phi[i]*prime[j];break;}
else phi[i*prime[j]]=phi[i]*(prime[j]-);
}
}
for(i=;i<=N;i++)
for(j=*i;j<=N;j+=i)
sum[j]+=i*phi[j/i];
for(i=;i<=N;i++) ans[i]=ans[i-]+sum[i];
while(){
scanf("%lld",&n);
if(n==) break;
printf("%lld\n",ans[n]);
}
return ;
}

https://www.luogu.org/problemnew/show/P2257

事实上此题做法与以上几题是一样的,如果数据范围一样。。

然而此题有多组数据又是1e7,需要更好的做法

不妨设n<=m

根据方法2,得到答案是$\sum_{1<=k<=n且k为质数}\sum_{i=1}^n\mu(i){\lfloor}\frac{n}{ik}{\rfloor}{\lfloor}\frac{m}{ik}{\rfloor}$

好像不能化简了?然而A不了题?

强行解释一波推算方法:注意到最终的可能求法是整除分块,要枚举ik的积,那么试着把它提出来

第二种解释方法:为什么要这么写式子?学莫比乌斯反演时学来的式子枚举的就是d啊

看了题解,发现可以令d=ik

答案就等于$\sum_{1<=k<=n且k为质数}\sum_{k|d}\mu(\frac{d}{k}){\lfloor}\frac{n}{d}{\rfloor}{\lfloor}\frac{m}{d}{\rfloor}$

$=\sum_{d=1}^n\sum_{k|d且k为质数}\mu(\frac{d}{k}){\lfloor}\frac{n}{d}{\rfloor}{\lfloor}\frac{m}{d}{\rfloor}$

$\sum_{k|d且k为质数}\mu(\frac{d}{k})$可以预处理了,根据那个质数个数(近似)公式,预处理复杂度大概接近O(n)了。。

可以A了。。。

 //#pragma GCC optimize("Ofast")
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<vector>
using namespace std;
#define fi first
#define se second
#define mp make_pair
#define pb push_back
typedef long long ll;
typedef unsigned long long ull;
typedef pair<int,int> pii;
#define N 10000000
int prime[N+],len,mu[N+],dd[N+];
bool nprime[N+];
int n,m;ll ans;
int main()
{
int i,j,k,T,TT;
mu[]=;
for(i=;i<=N;i++)
{
if(!nprime[i]) prime[++len]=i,mu[i]=-;
for(j=;j<=len&&i*prime[j]<=N;j++)
{
nprime[i*prime[j]]=;
if(i%prime[j]==) {mu[i*prime[j]]=;break;}
else mu[i*prime[j]]=-mu[i];
}
}
for(i=;i<=len;i++)
for(j=,k=prime[i];j<=N/prime[i];j++,k+=prime[i])
dd[k]+=mu[j];
for(i=;i<=N;i++) dd[i]+=dd[i-];
scanf("%d",&T);
for(TT=;TT<=T;TT++)
{
scanf("%d%d",&n,&m);
if(n>m) swap(n,m);
ans=;
for(i=;i<=n;i=j+)
{
j=min(n,min(n/(n/i),m/(m/i)));
ans+=ll(dd[j]-dd[i-])*(n/i)*(m/i);
}
printf("%lld\n",ans);
}
return ;
}

双倍经验https://www.luogu.org/problemnew/show/P2568

好吧以上这题时限有点紧。。不过单次询问O(n)也可以用方法3或者方法2

洛谷 P2398 GCD SUM || uva11417,uva11426,uva11424,洛谷P1390,洛谷P2257,洛谷P2568的更多相关文章

  1. 洛谷P2398 GCD SUM (数学)

    洛谷P2398 GCD SUM 题目描述 for i=1 to n for j=1 to n sum+=gcd(i,j) 给出n求sum. gcd(x,y)表示x,y的最大公约数. 输入输出格式 输入 ...

  2. 洛谷P2398 GCD SUM [数论,欧拉筛]

    题目传送门 GCD SUM 题目描述 for i=1 to n for j=1 to n sum+=gcd(i,j) 给出n求sum. gcd(x,y)表示x,y的最大公约数. 输入输出格式 输入格式 ...

  3. 洛谷P2398 GCD SUM

    题目描述 for i=1 to n for j=1 to n sum+=gcd(i,j) 给出n求sum. gcd(x,y)表示x,y的最大公约数. 输入输出格式 输入格式: n 输出格式: sum ...

  4. 洛谷 P2398 GCD SUM 题解

    题面 挺有意思的. 设f[i]表示gcd(i,j)=i的个数,g[i]表示k|gcd(i,j)的个数; g[i]=(n/i)*(n/i); g[i]=f[i]+f[2i]+f[3i]+...; 所以f ...

  5. P2398 GCD SUM

    P2398 GCD SUM一开始是憨打表,后来发现打多了,超过代码长度了.缩小之后是30分,和暴力一样.正解是,用f[k]表示gcd为k的一共有多少对.ans=sigma k(1->n) k*f ...

  6. *P2398 GCD SUM[数论]

    题目描述 for i=1 to n for j=1 to n sum+=gcd(i,j) 解析 给出n求sum. gcd(x,y)表示x,y的最大公约数. 直接枚举复杂度为\(O(n^2)\),显然无 ...

  7. acdream 1148 GCD SUM 莫比乌斯反演 ansx,ansy

    GCD SUM Time Limit: 8000/4000MS (Java/Others)Memory Limit: 128000/64000KB (Java/Others) SubmitStatis ...

  8. GCD SUM 强大的数论,容斥定理

    GCD SUM Time Limit: 8000/4000MS (Java/Others) Memory Limit: 128000/64000KB (Java/Others) SubmitStatu ...

  9. Luogu2398 GCD SUM

    Luogu2398 GCD SUM 求 \(\displaystyle\sum_{i=1}^n\sum_{j=1}^n\gcd(i,j)\) \(n\leq10^5\) 数论 先常规化式子(大雾 \[ ...

随机推荐

  1. 图像处理之图像格式变换和色彩增强---rgb2hsi2hsv 色彩增强

    从昨天折腾到今天.再折腾下去我都要上主楼了  大致和灰度图均衡是一样的,主要是不能像平滑什么的直接对R,G,B三个分量进行.这样出来的图像时没法看的.因此我们要对亮度进行均衡.而HSI彩色空间中的分量 ...

  2. OpenCV2.3.1在CentOS6.5下的安装

    安装的linux版本号是centos6.5.选择的是opencv2.3.1.不是非常新的版本号. 由于在安装opencv2.4.9的时候.make的过程中出现了问题. 一:安装依赖包 依赖包用yum安 ...

  3. hadoop 集群搭建 配置 spark yarn 对效率的提升永无止境

    [手动验证:任意2个节点间是否实现 双向 ssh免密登录] 弄懂通信原理和集群的容错性 任意2个节点间实现双向 ssh免密登录,默认在~目录下 [实现上步后,在其中任一节点安装\配置hadoop后,可 ...

  4. URL 下载

    package URL; import java.io.File;import java.io.FileOutputStream;import java.io.IOException;import j ...

  5. Java 深拷贝浅拷贝 与 序列化

    一.浅拷贝.深拷贝 浅拷贝会对对象中的成员变量进行拷贝:如果是基本类型,拷贝的就是基本类型的值:如果属性是内存地址(引用类型),拷贝的就是内存地址 : 深拷贝,除了基本类型外,引用类型所引用的对象也会 ...

  6. Delphi通过Get获取来自PHP的返回值

    Delphi代码 unit Unit1; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Contro ...

  7. 北斗有 35 颗卫星,而 GPS 有 24 颗卫星,为什么二者数量不同?

    作者:知乎用户链接:https://www.zhihu.com/question/21092045/answer/17164418来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载请注 ...

  8. 如何查看一个Application是32位的还是64位的?

    使用process explorer查看,找到对应的进程. 注册表的路径是Computer\HKEY_LOCAL_MACHINE\SOFTWARE\WOW6432Node\ 使用powershell查 ...

  9. java scoket客户端服务端发送消息

    客户端 public class User { public static void main(String[] args) { while (true) { try { Socket socket ...

  10. Android自定义控件实现带有清除按钮的EditText

    首先声明我也是参考了别人的思路,只是稍微做了下修改,增加显示密码与隐藏密码,没有输入字符串时让EditText进行抖动,废话少说这里附上效果图 效果很赞有木有 那么怎么实现这种效果呢?那就跟着我一起来 ...