【SPOJ-GCDEX】GCD Extreme(欧拉函数)
题目:
分析:
求:
\]
这道题给同届新生讲过,由于种种原因只讲了 \(O(n)\) 预处理欧拉函数 \(O(n)\) 查询的暴力做法,顺带提了一句 “这题能根号查询” 被教练嘴了 QAQ 。以及小恐龙给我说有 \(O(n\log n)\) 预处理 \(O(1)\) 查询的另一种写法。
重点是前几天某学长讲课讲这道题,才知道有 \(O(n)\) 预处理 \(O(1)\) 查询的神仙做法,并且据说因为 SPOJ 上时限 0.237s ,复杂度比这个高的都过不去(只交了最后一种,所以我也不知道是真的假的……)。下面分别介绍这三种做法(在无特殊说明的情况下,本文中所有除法均为向下取整):
方法一
为了方便计算,先给和式中添上 \(\sum_{i=1}^{n}gcd(i,i)=\frac{n(n+1)}{2}\),再换一下枚举顺序,得到:
\]
然后先枚举 \(gcd(i,j)\) 的值,并把 \(i\) 变成 \(i\cdot d\),\(j\) 变成 \(j\cdot d\) :
\]
(方括号表示其中式子成立时值为 \(1\) ,否则为 \(0\) )
根据欧拉函数的定义( \(\varphi(x)\) 表示不超过 \(x\) 的与 \(x\) 互质的正整数的个数),得到:
\]
(其中 \(S_\varphi\) 表示 \(\varphi\) 的前缀和,即 \(S_\varphi(x)=\sum_{i=1}^{x}\varphi(x)\) )
用线性筛处理出欧拉函数前缀和后数论分块即可。预处理 \(O(n)\) ,查询 \(O(\sqrt{n})\) 。
方法二
设:
\]
则答案为:
\]
考虑如何计算 \(F(i)\) 。同样先枚举 \(gcd(i,j)\) 的值:
\]
于是先线性筛 \(\varphi\) ,然后枚举 \(d\) ,每个 \(d\) 向所有 \(F(i\cdot d) (0<i\leq \frac{n}{d})\) 贡献 \(d\cdot \varphi(i)\) 。最后处理 \(F\) 函数的前缀和。预处理时间复杂度为 \(n\) 乘上 \(\sum_{i=1}^{n}\frac{1}{i}\),约为 \(O(n\log n)\) 。可以 \(O(1)\) 查询。
方法三:
接着方法二:
\]
则答案为:
\]
考虑如果已经求出了 \(F(n)\) ,则有(其中 \(p\) 是质数且与 \(n\) 互质):
\]
对于和式的每一项,这个 \(p\) 要么乘到 \(d\) 上,要么乘到 \(\varphi(i)\) 上,而欧拉函数是积性函数所以 \(\varphi(ip)=p\cdot \varphi(i)\) 。
这个性质很好,如果能快速计算 \(n=p^k\) (\(p\) 是质数)的情况就能愉快地线性筛了。(为什么?请看 【知识总结】线性筛_杜教筛_Min25筛 的线性筛部分)
考虑当 \(n=p^k\) ,\(F(n)=\sum_{i=0}^{k}p^{k-i}\varphi(p^i)\) 。根据欧拉函数的公式:
\]
(注意当 \(i=0\) 时 \(p^0=1\) 中没有因子 \(p\) ,要特殊处理)
所以当 \(k>1\) , \(F(p^{k-1})=(k-1)p^{k-2}(p-1)+p^{k-1}\) ,给它乘上 \(p\) 再加上 \(p^{k-1}(p-1)\) 就是 \(F(p^k)\) 。 \(k=1\) (即 \(n\) 是质数)的情况直接手算一下,等于 \(2n-1\) 。
于是可以线性筛出 \(F\) 函数然后处理前缀和, \(O(n)\) 预处理 \(O(1)\) 查询。
代码:
(方法三)
#include <cstdio>
#include <cctype>
#include <cstring>
#include <algorithm>
using namespace std;
namespace zyt
{
template<typename T>
inline bool read(T &x)
{
char c;
bool f = false;
x = 0;
do
c = getchar();
while (c != EOF && c != '-' && !isdigit(c));
if (c == EOF)
return false;
if (c == '-')
f = true, c = getchar();
do
x = x * 10 + c - '0', c = getchar();
while (isdigit(c));
if (f)
x = -x;
return true;
}
template<typename T>
inline void write(T x)
{
static char buf[20];
char *pos = buf;
if (x < 0)
putchar('-'), x = -x;
do
*pos++ = x % 10 + '0';
while (x /= 10);
while (pos > buf)
putchar(*--pos);
}
typedef long long ll;
const int N = 1e6 + 10;
ll f[N];
int prime[N], last[N], cnt;
bool mark[N];
void init()
{
f[1] = 1;
for (int i = 2; i < N; i++)
{
if (!mark[i])
prime[cnt++] = last[i] = i, f[i] = i * 2 - 1;
for (int j = 0; j < cnt && (ll)i * prime[j] < N; j++)
{
int k = i * prime[j];
mark[k] = true;
if (i % prime[j] == 0)
{
last[k] = last[i] * prime[j];
if (last[k] == k)
f[k] = f[i] * prime[j] + (ll)i * (prime[j] - 1);
else
f[k] = f[i / last[i]] * f[last[k]];
}
else
{
last[k] = prime[j];
f[k] = f[i] * f[prime[j]];
}
}
}
for (int i = 1; i < N; i++)
f[i] += f[i - 1];
}
int work()
{
int n;
init();
while (read(n) && n)
write(f[n] - (ll)(n + 1) * n / 2), putchar('\n');
return 0;
}
}
int main()
{
return zyt::work();
}
【SPOJ-GCDEX】GCD Extreme(欧拉函数)的更多相关文章
- spoj 3871. GCD Extreme 欧拉+积性函数
3871. GCD Extreme Problem code: GCDEX Given the value of N, you will have to find the value of G. Th ...
- hdu2588 GCD (欧拉函数)
GCD 题意:输入N,M(2<=N<=1000000000, 1<=M<=N), 设1<=X<=N,求使gcd(X,N)>=M的X的个数. (文末有题) 知 ...
- uva11426 gcd、欧拉函数
题意:给出N,求所有满足i<j<=N的gcd(i,j)之和 这题去年做过一次... 设f(n)=gcd(1,n)+gcd(2,n)+......+gcd(n-1,n),那么answer=S ...
- HDU 1695 GCD (欧拉函数+容斥原理)
GCD Time Limit: 6000/3000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submiss ...
- HDU 1787 GCD Again(欧拉函数,水题)
GCD Again Time Limit: 1000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total S ...
- hdu 4983 Goffi and GCD(欧拉函数)
Problem Description Goffi is doing his math homework and he finds an equality on his text book: gcd( ...
- hdu 1695 GCD(欧拉函数+容斥)
Problem Description Given 5 integers: a, b, c, d, k, you're to find x in a...b, y in c...d that GCD( ...
- HDU 1695 GCD(欧拉函数+容斥原理)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1695 题意:x位于区间[a, b],y位于区间[c, d],求满足GCD(x, y) = k的(x, ...
- GCD(欧拉函数)
GCD Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Submissio ...
- HDU 2588 GCD(欧拉函数)
GCD Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Submis ...
随机推荐
- 洛谷—— P2543 [AHOI2004]奇怪的字符串
P2543 [AHOI2004]奇怪的字符串 题目描述 输入输出格式 输入格式: 输入文件中包含两个字符串X和Y.当中两字符串非0即1.序列长度均小于9999. 输出格式: X和Y的最长公共子序列长度 ...
- 利用Python爬虫实现百度网盘自动化添加资源
事情的起因是这样的,由于我想找几部经典电影欣赏欣赏,于是便向某老司机寻求资源(我备注了需要正规视频,绝对不是他想的那种资源),然后他丢给了我一个视频资源网站,说是比较有名的视频资源网站.我信以为真,便 ...
- c++11中的线程、锁和条件变量
void func(int i, double d, const string& s) { cout << i << ", " << d ...
- 介绍css 的3D 变换(3D transform)
https://desandro.github.io/3dtransforms/docs/card-flip.html ---------------------------------------- ...
- 使用OpenCV读、操作、写图像并与bash合作对某个文件夹下全部图像进行相似处理
我门要对某个文件夹下全部图像文件进行统一处理,假设图像的数量过多.那么手动地一张张处理就会显得有些麻烦.本文使用OpenCV和bash来完毕我们指定的任务. 任务 将文件夹A下的全部统一格式的jpg图 ...
- Android GMS无法通过网络定位
前言 欢迎大家我分享和推荐好用的代码段~~ 声明 欢迎转载.但请保留文章原始出处: CSDN:http://www.csdn.net ...
- 用Java开发50个棋类游戏
眼下已经开发完了两个 1A2B 24点 打算开发以下的.直接在QQ上玩. QQ机器人已经有了.我们直接写业务即可.有兴趣的參与.机器人婷婷体验群 Java技术交流 207224939 四棋 小枪大炮 ...
- (十)Net Core项目使用Cookies (八)Net Core项目使用Controller之三-入参
(十)Net Core项目使用Cookies 一.简介 1.Net Core可以直接使用Cookies,但是调用方式有些区别. 2.Net Core将Request和Response分开实现. 二.基 ...
- Android ListView的item点击无响应的解决方法
假设listitem里面包含button或者checkbox等控件,默认情况下listitem会失去焦点,导致无法响应item的事件,最经常使用的解决的方法 是在listitem的布局文件里设置des ...
- 人脸和性别识别(基于OpenCV)
描写叙述 人脸识别包含四个步骤 人脸检測:定位人脸区域,仅仅关心是不是脸: 人脸预处理:对人脸检測出来的图片进行调整优化. 收集和学习人脸:收集要识别的人的预处理过的人脸,然后通过一些算法去学习怎样识 ...