51nod - 1188 - 最大公约数之和 V2 - 数论
https://www.51nod.com/Challenge/Problem.html#!#problemId=1188
求\(\sum\limits_{i=1}^{n-1}\sum\limits_{j=i+1}^{n}gcd(i,j)\)
首先交换求和\(\sum\limits_{j=2}^{n}\sum\limits_{i=1}^{j-1}gcd(i,j)=\sum\limits_{j=2}^{n}\sum\limits_{i=1}^{j}gcd(i,j)-j\)
像之前那样用莫比乌斯反演搞出
\(\sum\limits_{g=1}^{n}g\sum\limits_{i=1}^{\lfloor\frac{n}{g}\rfloor}{\varphi(i)}, (\varphi(1)=0)\)
或者
\(\sum\limits_{g=1}^{n}g\sum\limits_{i=1}^{\lfloor\frac{n}{g}\rfloor}{\mu(i)}*{\lfloor\frac{n}{i*g}\rfloor}^2\)
都是T了,说明单次回答的复杂度非常感人,原因是因为跟n有关就逃不了分块。
所以分块的复杂度还是太高了。
然后惊人的事情发生了。
这道题和前面的不一样的地方在于,他的n其实没有前面的大,只是多!
分块的意义在于加快单次回答的速度,所以才把g提了出来分块,现在不太合适。
正解,先求解:
\(\sum\limits_{i=1}^{n}gcd(i,n)\)
这个东西可以枚举g:
\(\sum\limits_{g=1}^{n}g\sum\limits_{i=1}^{n}[gcd(i,n)==g]\)
\(\sum\limits_{g=1}^{n}g\sum\limits_{i=1}^{\lfloor\frac{n}{g}\rfloor}[gcd(i,\lfloor\frac{n}{g}\rfloor)==1]\)
\(\sum\limits_{g=1}^{n}g\varphi(\lfloor\frac{n}{g}\rfloor)\)
这个有什么不对呢?其实没什么不对,就是不好。因为gcd必定是整除n的,所以可以:
\(\sum\limits_{g|n}g\varphi(\frac{n}{g})\)
这明显就是一个积性函数,记为 \(p(n)\) ,处理好就可以了。原式:
\(\sum\limits_{j=2}^{n}p(j)-j\)
这个式子 \(p(n)\) 用埃筛已经足够了,因为是5e6的数据量,1400ms/2000ms跑过去了。但是我在想会不会有线性的筛法。
2019/6/7早上试了很多种之后,发现还是直接利用积性函数的结论最简单,对于每个合数记录他的最小质因子及其幂次,然后一除就得到两个互质的因数。
问题在于纯p的幂次一除就变成没有了。也就是别人的博客说的 \(p^k\) 要特殊处理,想到上面这个式子是一个狄利克雷卷积,对于纯p的幂次只要从他的低1次转移过来就可以了。
最后套上快读快写狗掉榜首!需要注意的是我准备的线性筛模板里多用了0.25倍空间,其实真的可以充分利用的。比如这个的话pk肯定不是0,那么就用pk来做判断就可以了。
所以说的确是空间换时间,多了0.5倍的空间省去一个logn倍。200ms/2000ms稳得一笔。
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
inline int read() {
int x=0;
char c=getchar();
while(c<'0'||c>'9')
c=getchar();
do {
x=(x<<3)+(x<<1)+c-'0';
c=getchar();
} while(c>='0'&&c<='9');
return x;
}
inline void write(ll x) {
//printf("%lld\n",x);
if(x>9) {
write(x/10);
}
putchar(x%10+'0');
return;
}
const int MAXN=5e6;
int pri[MAXN+1];
int &pritop=pri[0];
ll ans[MAXN+1];
int pk[MAXN+1];//pk(n)=pk的最小因子p的次方
void sieve(int n=MAXN) {
ans[1]=1;
pk[1]=1;
for(int i=2; i<=n; i++) {
if(!pk[i]) {
pri[++pritop]=i;
ans[i]=i+i-1;
pk[i]=i;
}
for(int j=1; j<=pritop&&i*pri[j]<=n; j++) {
int t=i*pri[j];
if(i%pri[j]) {
pk[t]=pk[pri[j]];
ans[t]=ans[i]*ans[pri[j]];
} else {
pk[t]=pk[i]*pri[j];
if(pk[t]==t) {
//纯p的幂次,没有其他因子,这样算不会漏因子
//t-t/pri[j]就是phi[t]
ans[t]=(t-t/pri[j])+ans[i]*pri[j];
} else {
//积性函数
ans[t]=ans[pk[t]]*(ans[t/pk[t]]);
}
break;
}
}
}
for(int i=1; i<=n; i++)
ans[i]+=ans[i-1]-i;
}
inline void solve() {
sieve();
int T=read();
while(T--) {
int n=read();
write(ans[n]);
putchar('\n');
}
}
int main() {
#ifdef Yinku
freopen("Yinku.in","r",stdin);
#endif // Yinku
solve();
return 0;
}
其他尝试?
现在把g放回去。
\(\sum\limits_{g=1}^{n}\sum\limits_{i=1}^{\lfloor\frac{n}{g}\rfloor}g*{\varphi(i)}, (\varphi(1)=0)\)
这个不就是相当于把每个 \(i*g \leq n\) 的加起来吗?所以当然可以交换求和顺序!
\(\sum\limits_{i=1}^{n}\sum\limits_{g=1}^{\lfloor\frac{n}{i}\rfloor}g*{\varphi(i)}, (\varphi(1)=0)\)
前面是一个干净的前缀和,后面那个式子是nlogn的!
也就是预处理:
\(ans(n)=\sum\limits_{g=1}^{\lfloor\frac{n}{i}\rfloor}g*{\varphi(i)}, (\varphi(1)=0)\)
51nod - 1188 - 最大公约数之和 V2 - 数论的更多相关文章
- 51nod 1188 最大公约数之和 V2
第二个\( O(T\sqrt(n)) \)复杂度T了..T了..T了...天地良心,这能差多少?! 于是跑去现算(. \[ \sum_{i=1}^{n-1}\sum_{j=i+1}^{n}gcd(i, ...
- 51 nod 1188 最大公约数之和 V2
1188 最大公约数之和 V2 题目来源: UVA 基准时间限制:2 秒 空间限制:262144 KB 分值: 160 难度:6级算法题 给出一个数N,输出小于等于N的所有数,两两之间的最大公约数 ...
- 1188 最大公约数之和 V2
1188 最大公约数之和 V2 题目来源: UVA 基准时间限制:2 秒 空间限制:262144 KB 给出一个数N,输出小于等于N的所有数,两两之间的最大公约数之和. 相当于计算这段程 ...
- 51nod 1040:最大公约数之和(数论)
题目链接:https://www.51nod.com/onlineJudge/questionCode.html#!problemId=1040 给出一个n,求1-n这n个数,同n的最大公约数的和. ...
- 51nod1188 最大公约数之和 V2
考虑每一个数对于答案的贡献.复杂度是O(nlogn)的.因为1/1+1/2+1/3+1/4......是logn级别的 //gcd(i,j)=2=>gcd(i/2,j/2)=1=>phi( ...
- 51nod 1040 最大公约数之和(欧拉函数)
1040 最大公约数之和 题目来源: rihkddd 基准时间限制:1 秒 空间限制:131072 KB 分值: 80 难度:5级算法题 给出一个n,求1-n这n个数,同n的最大公约数的和.比如: ...
- 51nod 1237 最大公约数之和 V3(杜教筛)
[题目链接] https://www.51nod.com/onlineJudge/questionCode.html#!problemId=1237 [题目大意] 求[1,n][1,n]最大公约数之和 ...
- 51nod 1040 最大公约数之和 欧拉函数
1040 最大公约数之和 题目连接: https://www.51nod.com/onlineJudge/questionCode.html#!problemId=1040 Description 给 ...
- 51NOD 1237 最大公约数之和 V3 [杜教筛]
1237 最大公约数之和 V3 题意:求\(\sum_{i=1}^n\sum_{j=1}^n(i,j)\) 令\(A(n)=\sum_{i=1}^n(n,i) = \sum_{d\mid n}d \c ...
随机推荐
- Pycharm下HTMLTestRunner不生成测试报告
网上搜索资料,最终找到了本次解决的方案: 1.修改Edit Configurations... 2.将测试脚本从Python tests中删除,再Python下新增脚本,这样就不会运行自带的unitt ...
- 实记处理mongodb的NUMA问题
一次在启动mongodb的过程中,出现过NUMA这个问题, mongodb日志显示如下: WARNING: You are running on a NUMA machine. We suggest ...
- C#高级编程七十五天----C#使用指针
在C#中使用指针的语法 假设想在C#中使用指针,首先对项目进行过配置: watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQv/font/5a6L5L2T/font ...
- android-async-http框架
android-async-http 简单介绍:An asynchronous, callback-based Http client for Android built on top of Apac ...
- EasyDarwin流媒体云平台:EasyCamera开源摄像机接入海康威视摄像机实时视频
本文转自EasyDarwin团队成员Alex的博客:http://blog.csdn.net/cai6811376/article/details/52755298 EasyCamera接收云平台实时 ...
- 嵌套的EasyUI 怎么获取对象
说明: 1.本篇文章介绍的是,怎么获取嵌套的Easyui 中的id为pageDetail的iframe对象 2.刚开始的页面效果如下图,是一个只有north,center区域的easyUI easy ...
- Please read "Security" section of the manual to find out how to run mysqld as root!
[root@test ~]# /usr/local/mysql/bin/mysqld2018-08-05T08:29:05.143142Z 0 [Warning] [MY-011070] [Serve ...
- dokcer3
安装好的文件位置: /usr/sbin/nginx:主程序 /etc/nginx:存放配置文件 /usr/share/nginx:存放静态文件 /var/log/nginx:存放日志 其实从上面的根目 ...
- 异步模式模式Future(结合Callable可以获取线程返回结果)
submit 和 excute是有啥区别 如果有这样的需求: 多线程实现下载,提高效率. 不论是Thread类还是Runnable接口重写run方法,有个特点就是没有返回值~~~~~~ 我都主线程 如 ...
- 异步执行js脚本——防止阻塞
JS允许我们修改页面中的所有方面:内容,样式和用户进行交互时的行为. 但是js同样可以阻塞DOM树的形成并且延迟页面的渲染. 让你的js变成异步执行,并且减少不必要的js文件从而提高性能. JavaS ...