http://acm.hdu.edu.cn/showproblem.php?pid=6588

新学到了一个求n以内与m的gcd的和的快速求法。也就是下面的S1。


①求:

$ \sum\limits_{i=1}^{n}gcd(m,i) $

②枚举d:

$ \sum\limits_{d|m} d \sum\limits_{i=1}^{n} [gcd(m,i)==d] $

③显然:

$ \sum\limits_{d|m} d \sum\limits_{i=1}^{\lfloor\frac{n}{d}\rfloor} [gcd(\frac{m}{d},i)==1] $

到这一步已经可以递归求了,琪琪说是 \(O(n^{\frac{3}{4}})\) ,不过题解可以继续往下。

④为了方便直接考虑 $ \sum\limits_{i=1}^{n} [gcd(m,i)==1] $ ,反演(大概):

$ \sum\limits_{i=1}^{n} \sum\limits_{d|gcd(m,i)} \mu(d) $

⑤交换一下顺序,枚举d,很显然n以内的d的倍数都会贡献一个mu(d):

$ \sum\limits_{d|m} \mu(d) \lfloor\frac{n}{d}\rfloor $


下面的是根据题解的实现,不过是__int64的版本。

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef __int64 lll; const int mod = 998244353;
const int MAXN = 10000000; int phi[MAXN + 1];
int pri[MAXN + 1], pritop;
bool notpri[MAXN + 1]; void sieve() {
int n = MAXN;
pri[1] = phi[1] = 1;
for(int i = 2; i <= n; i++) {
if(!pri[i])
pri[++pritop] = i, phi[i] = i - 1;
for(int j = 1, tmp; j <= pritop && (tmp = i * pri[j]) <= n; j++) {
pri[tmp] = 1;
if(i % pri[j])
phi[tmp] = phi[i] * phi[pri[j]];
else {
phi[tmp] = phi[i] * pri[j];
break;
}
}
}
} ll S1(lll n, int m) {
//sigma gcd(i,m) [1,n]
ll res = 0;
for(int T = 1; T * T <= m; ++T) {
if(!(m % T)) {
res += (n / T) * phi[T];
if(T * T != m) {
res += (n / (m / T)) * phi[(m / T)];
}
}
}
res %= mod;
return res;
} ll qpow(ll x, int n) {
ll res = 1;
while(n) {
if(n & 1)
res = res * x % mod;
x = x * x % mod;
n >>= 1;
}
return res;
} const int inv2 = qpow(2ll, mod - 2);
const int inv6 = qpow(6ll, mod - 2); ll sigma1(ll x) {
return x * (x + 1ll) % mod * inv2 % mod;
} ll sigma2(ll x) {
return x * (x + 1ll) % mod * (2ll * x + 1ll) % mod * inv6 % mod;
} ll S2_1(int r, int T) {
int c = r / T;
ll res = 0;
res += 3ll * T * sigma2(c);
res += 3ll * sigma1(c);
res += c;
res %= mod;
return res;
} ll S2(int r) {
ll res = 0;
for(int T = 1; T <= r; ++T) {
res += 1ll * phi[T] * S2_1(r, T) % mod;
}
res %= mod;
return res;
} ll S0(lll n) {
lll i, i3;
for(i = 1;; ++i) {
lll tmp = i * i * i;
if(tmp > n) {
--i;
break;
} else
i3 = tmp;
}
ll res = 0;
res += S1(n, i) - S1(i3 - 1, i);
res += S2(i - 1);
res = (res % mod + mod) % mod;
return res;
} inline lll read() {
lll x = 0;
char c;
do {
c = getchar();
} while(c < '0' || c > '9');
do {
x = (x << 3) + (x << 1) + c - '0';
c = getchar();
} while(c >= '0' && c <= '9');
return x;
} int main() {
#ifdef Yinku
freopen("Yinku.in", "r", stdin);
#endif // Yinku
sieve();
int T;
cin >> T;
lll n;
while(T--) {
n = read();
cout << S0(n) << endl;
}
}

其实具体的思路还是要先分成两部分来算,但是我当时不会计算这个S1导致T了。其实计算S2的时候在整数较大的时候发生了溢出。也就是c*c的位置。所以说以后非数组的值一律开ll就对了。

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef __int128 lll; const int mod = 998244353;
const int MAXN = 10000000; int pk[MAXN + 1];
int sum1[MAXN + 1];
int phi[MAXN + 1];
int pri[MAXN + 1], pritop;
bool notpri[MAXN + 1]; void sieve() {
int n = MAXN;
pri[1] = pk[1] = sum1[1] = phi[1] = 1;
for(int i = 2; i <= n; i++) {
if(!pri[i])
pri[++pritop] = i, phi[i] = i - 1, pk[i] = i, sum1[i] = 2 * i - 1;
for(int j = 1, p, tmp; j <= pritop && (p = pri[j]) && (tmp = i * p) <= n; j++) {
pri[tmp] = 1;
if(i % p) {
pk[tmp] = pk[p];
sum1[tmp] = 1ll * sum1[i] * sum1[p] % mod;
phi[tmp] = phi[i] * phi[p];
} else {
pk[tmp] = pk[i] * p;
if(pk[tmp] == tmp) {
sum1[tmp] = (1ll * sum1[i] * p % mod + (tmp - tmp / p)) % mod;
} else {
sum1[tmp] = 1ll * sum1[pk[tmp]] * sum1[tmp / pk[tmp]] % mod;
}
phi[tmp] = phi[i] * p;
break;
}
}
}
} int sum2[MAXN + 1]; ll qpow(ll x, int n) {
ll res = 1;
while(n) {
if(n & 1)
res = res * x % mod;
x = x * x % mod;
n >>= 1;
}
return res;
} void init() {
for(int c = 1, c1 = 2, c2 = 7, f1 = 1; c <= MAXN;) {
sum2[c] = ((1ll * c2 - f1 + mod) % mod * sum1[c] % mod * qpow(c, mod - 2) % mod + c + sum2[c - 1]) % mod;
++c, ++c1, f1 = c2 + 1;
c2 = (1ll * c1 * c1 % mod * c1 % mod - 1 + mod) % mod;
}
} ll S1(lll n, int m) {
//sigma gcd(i,m) [1,n]
ll res = 0;
for(int T = 1; T * T <= m; ++T) {
if(!(m % T)) {
res += (n / T) * phi[T];
if(T * T != m) {
res += (n / (m / T)) * phi[(m / T)];
}
}
}
res %= mod;
return res;
} ll S0(lll n) {
lll i, i3;
for(i = 1;; ++i) {
lll tmp = i * i * i;
if(tmp > n) {
--i;
break;
} else
i3 = tmp;
}
ll res = 0;
res += S1(n, i) - S1(i3 - 1, i);
res += sum2[i - 1];
res = (res % mod + mod) % mod;
return res;
} inline lll read() {
lll x = 0;
char c;
do {
c = getchar();
} while(c < '0' || c > '9');
do {
x = (x << 3) + (x << 1) + c - '0';
c = getchar();
} while(c >= '0' && c <= '9');
return x;
} int main() {
#ifdef Yinku
freopen("Yinku.in", "r", stdin);
#endif // Yinku
sieve();
init();
int T;
cin >> T;
lll n;
while(T--) {
n = read();
cout << S0(n) << endl;
}
}

2019 Multi-University Training Contest 1 - 1011 - Function - 数论的更多相关文章

  1. 2014 Multi-University Training Contest 1/HDU4861_Couple doubi(数论/法)

    解题报告 两人轮流取球,大的人赢,,, 贴官方题解,,,反正我看不懂.,,先留着理解 关于费马小定理 关于原根 找规律找到的,,,sad,,, 非常easy找到循环节为p-1,每个循环节中有一个非零的 ...

  2. 2019 Multi-University Training Contest 2 - 1011 - Keen On Everything But Triangle - 线段树

    http://acm.hdu.edu.cn/showproblem.php?pid=6601 首先要贪心地想,题目要最长的边长,那么要怎么构造呢?在一段连续的区间里面,一定是拿出最长的三根出来比,这样 ...

  3. 2016 Multi-University Training Contest 5 1011 Two DP

    http://acm.hdu.edu.cn/showproblem.php?pid=5791 HDU5791 Two 题意 :两个数组,多少个不连续子串相等 思路: dp[i][j] :a串i结尾,b ...

  4. HDU 6342.Problem K. Expression in Memories-模拟-巴科斯范式填充 (2018 Multi-University Training Contest 4 1011)

    6342.Problem K. Expression in Memories 这个题就是把?变成其他的使得多项式成立并且没有前导零 官方题解: 没意思,好想咸鱼,直接贴一篇别人的博客,写的很好,比我的 ...

  5. 2016 Multi-University Training Contest 3 1011【鸽巢原理】

    题解: 坐标(0,m)的话,闭区间,可能一共有多少曼哈顿距离? 2m 但是给一个n,可能存在n(n+1)/2个曼哈顿距离 所以可以用抽屉原理了 当n比抽屉的数量大,直接输出yes 不用计算 那...N ...

  6. 2019 Nowcoder Multi-University Training Contest 4 E Explorer

    线段树分治. 把size看成时间,相当于时间 $l$ 加入这条边,时间 $r+1$ 删除这条边. 注意把左右端点的关系. #include <bits/stdc++.h> ; int X[ ...

  7. 2019 Nowcoder Multi-University Training Contest 1 H-XOR

    由于每个元素贡献是线性的,那么等价于求每个元素出现在多少个异或和为$0$的子集内.因为是任意元素可以去异或,那么自然想到线性基.先对整个集合A求一遍线性基,设为$R$,假设$R$中元素个数为$r$,那 ...

  8. 2019杭电多校一 K. Function (数论)

    大意: 给定$n(n\le 10^{21})$, 求$\sum\limits_{i=1}^n gcd(\lfloor\sqrt[3]{i}\rfloor,i)\mod 998244353$ 首先立方根 ...

  9. 2015 Multi-University Training Contest 8 hdu 5390 tree

    tree Time Limit: 8000ms Memory Limit: 262144KB This problem will be judged on HDU. Original ID: 5390 ...

随机推荐

  1. VS 2019编辑含有资源文件.resx的项目时提示MSB3086 任务未能使用 SdkToolsPath 或注册表项找到“al.exe”

    环境: Win10 X64, VS2019 错误提示: 错误 MSB3086 任务未能使用 SdkToolsPath“”或注册表项“HKEY_LOCAL_MACHINE\SOFTWARE\Micros ...

  2. 82-基于Xilinx Virtex-5LXT FPGA的四路光纤PCIE卡(4路光纤卡)

    基于Xilinx Virtex-5LXT FPGA的四路光纤PCIE卡(4路光纤卡) 1.板卡特点: 1)  主芯片采用Xilinx Virtex-5LXT FPGA. 前面板提供4路光纤接口,速率可 ...

  3. window环境下mysql导入sql文件时报错:ERROR: ASCII '\0' appeared in the statement

    错误信息: ERROR: ASCII '\0' appeared in the statement, but this is not allowed unless option --binary-mo ...

  4. [python 学习] 编码

    一.源文件编码(encoding: utf-8) 1. python 2.x 默认按ascii编码读取源文件,源码中出现了ascii不能表示的字符 "的",所以报错(3.x版本不报 ...

  5. windowserver 常用命令

    1.查看端口占用: netstat -ano | findstr "服务端口号"2.查看程序运行id: tasklist | findstr  nginx 3.杀死进程  tskk ...

  6. vue-cli 2.0搭建vue脚手架步骤

    1.安装node 检测版本node -v 2.安装webpack npm install webpack -g 检测版本 webpack -v 3.安装vue-cli npm install vue- ...

  7. 神仙dcx出的一道题

    题目大意 \(\;\;\)在一个坐标系上, 以\((0, 0)\)为起点, 每走一步,可以从\((x,y)\)走到\((x+1,y),(x-1,y),(x,y+1),(x,y-1)\)中的一个点上, ...

  8. C++ decltype

    #include <iostream> using namespace std; int main() { int ia{3}; decltype(ia) varr[3]={1,2,3}; ...

  9. centos启动提示unexpected inconsistency RUN fsck MANUALLY

    今天一台虚拟机背后的物理机故障了,主机迁移后变成了 read only filesystem.上面部署了很多长连接服务,没有关掉就直接reboot,报错: unexpected inconsisten ...

  10. python 全栈开发,Day45(html介绍和head标签,body标签中相关标签)

    一.html介绍 1.web标准 web准备介绍: w3c:万维网联盟组织,用来指定web标准的机构(组织) web标准:制作网页遵循的规范 web准备规范的分类:结构标准.表现标准.行为标准. 结构 ...