洛谷 P2398 GCD SUM || uva11417,uva11426,uva11424,洛谷P1390,洛谷P2257,洛谷P2568
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的更多相关文章
- 洛谷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的最大公约数. 输入输出格式 输入 ...
- 洛谷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的最大公约数. 输入输出格式 输入格式 ...
- 洛谷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 ...
- 洛谷 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 ...
- P2398 GCD SUM
P2398 GCD SUM一开始是憨打表,后来发现打多了,超过代码长度了.缩小之后是30分,和暴力一样.正解是,用f[k]表示gcd为k的一共有多少对.ans=sigma k(1->n) k*f ...
- *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)\),显然无 ...
- acdream 1148 GCD SUM 莫比乌斯反演 ansx,ansy
GCD SUM Time Limit: 8000/4000MS (Java/Others)Memory Limit: 128000/64000KB (Java/Others) SubmitStatis ...
- GCD SUM 强大的数论,容斥定理
GCD SUM Time Limit: 8000/4000MS (Java/Others) Memory Limit: 128000/64000KB (Java/Others) SubmitStatu ...
- Luogu2398 GCD SUM
Luogu2398 GCD SUM 求 \(\displaystyle\sum_{i=1}^n\sum_{j=1}^n\gcd(i,j)\) \(n\leq10^5\) 数论 先常规化式子(大雾 \[ ...
随机推荐
- directdraw 显示yuv
http://www.cnblogs.com/lidan/archive/2012/03/23/2413772.html http://www.yirendai.com/msd/
- POJ 2886 Who Gets the Most Candies?(树状数组+二分)
题目链接 注意题目中给的顺序是顺时针的,所以在数组中应该是倒着存的.左就是顺时针,右就是逆时针.各种调试之后,终于A了,很多种情况考虑情况. #include <cstring> #inc ...
- armel、armhf和arm64
1 这些名词是什么的缩写 1.1 armel 是arm eabi little endian的缩写.eabi是软浮点二进制接口,这里的e是embeded,是对于嵌入式设备而言. 1.2 armhf 是 ...
- Java客户端:调用EyeKey HTTP接口进行人脸对比
package com.example.buyishi; import java.io.BufferedReader; import java.io.IOException; import java. ...
- gravity layout_gravity
gravity:控制当前视图的内容/子view layout_gravity:控制视图本身
- js-easyUI格式化时间
formatter : function(value, row) { if(value != null){ var date = new Date(value); var y = date.getFu ...
- A喝酒(北京林业大学校赛)
http://www.jisuanke.com/contest/1410 王大钉喜欢喝酒,存货都喝完了,他就去楼下买,正好楼下的商店为了响应学校的 ACM 校赛推出了优惠活动:凡是在本店买的啤酒,喝完 ...
- i2c_set_clientdata函数【转】
本文转载自‘:http://blog.csdn.net/jk198310/article/details/43738367 在i2c驱动中有很多函数和数据结构,很多一时难以理解,所以写下本文共同学习. ...
- fork函数的使用【学习笔记】
#include "apue.h" ; char buf[] = "a write to stdout\r\n"; int main(void) { int v ...
- HDU3507 Print Article —— 斜率优化DP
题目链接:https://vjudge.net/problem/HDU-3507 Print Article Time Limit: 9000/3000 MS (Java/Others) Mem ...