1 条题解
-
0梁老师 (liangjianken) LV 8 SU @ 2024-7-19 9:36:34
题解
题意简述
给定 B,多次求 \sum\limits_{i=L}^R \varphi^{(\max_{j=1}^i \varphi(j) - B)}(i)。
日记不会这种科技,数据怎么造的? 你不会真的认为我写的是暴力吧 /mm
知识点
质因数分解、欧拉函数、(可选)线性筛
解题思路
前置知识
为解决此题,我们首先要证明一些东西。
对一个正整数 ,满足 的最小正整数 不超过 。
证明:分类讨论。
- 当 时,根据 的定义式,所有形如 的数均不与 互质。因此,。
- 当 且 时,。因此,。
额外地,可以证明奇数的 函数一定为偶数。证明:若 与 互质, 也与 互质。特例在 处,而奇数不存在这一处。
综上,每两次迭代, 必然缩小到 。因此,
- 事实上,可以进一步地缩小下界到 ,但此题不需要。
对一个正整数 ,。
证明:,因此迭代次数非正。
对一个正整数 ,,其中 为一个较小常数(可视为 )。
这并非一个严谨证明。考虑最小的 (其中 为一个更小的常数,可视为 ),则对一个正整数 ,有 ,因此迭代次数至少为 。由于 足够大(),迭代结果一定为 (见第一条证明)。
下一步目标,是找到 的上界。它相当于两个相邻质数差的最大值。万幸, 范围内,这一最大值不超过 。(日记:暴力跑的)
完整做法
以上已经把题说得差不多了。我们已经对 和 的情况给出了能够达到 的做法。
剩余的数一共有 个。考虑暴力求。利用质因数分解求 值是容易做到 的。而迭代次数是 的,这部分的复杂度为 。
而 固定时,这一部分是不变的。因此只需要一次预处理。总复杂度:。
诈骗:答案显然不会超过 ,因此开个
long long
就行。以上 均指 ,即数据范围的第三列值。
标准代码
C++
#include <bits/stdc++.h> using namespace std; #define int long long bool inp[1000005]; int pri[1000005], pc; void sieve(int mx) { inp[0] = inp[1] = true; for (int i = 2; i <= mx; ++i) { if (!inp[i]) pri[++pc] = i; for (int j = 1; i * pri[j] <= mx && j <= pc; ++j) { inp[i * pri[j]] = true; if (i % pri[j] == 0) break; } } return; } bool is_prime(int x) { if (x <= 1) return false; for (int i = 1; pri[i] * pri[i] <= x; ++i) { if (x % pri[i] == 0) return false; } return true; } int phi(int x) { int ans = x; for (int i = 1; pri[i] * pri[i] <= x; ++i) { if (x % pri[i] == 0) { while (x % pri[i] == 0) { x /= pri[i]; } ans = ans / pri[i] * (pri[i] - 1); } } if (x > 1) ans = ans / x * (x - 1); return ans; } int T, B; int f(int x, int max_phi) { max_phi -= B; if (max_phi <= 0) return x; while (max_phi--) { x = phi(x); if (x == 1) return 1; } return x; } int p1, p2; int val[1000005], ps[1000005]; int a(int x) { return x * (x + 1) / 2; // 1 + 2 + ... + x } int b(int x) { return ps[x - p1]; } int c(int x) { return x - p2; // p2+1 p2+2 .. x } void pre() { for (int i = B; i; --i) { if (is_prime(i)) { p1 = i; break; } } for (int i = B + 60; i <= 1100000000; ++i) { if (is_prime(i)) { p2 = i; break; } } int mx = phi(p1); for (int i = p1; i <= p2; ++i) { mx = max(mx, phi(i)); val[i - p1] = f(i, mx); if (i != p1) ps[i - p1] = ps[i - p1 - 1] + val[i - p1]; else ps[i - p1] = val[i - p1]; } } int solve(int x) { if (x < p1) return a(x); else if (x <= p2) return a(p1 - 1) + b(x); else return a(p1 - 1) + b(p2) + c(x); } signed main() { //freopen("phi.in", "r", stdin); //freopen("phi.out", "w", stdout); ios::sync_with_stdio(false); cin.tie(0); cout.tie(0); sieve(1000000); cin >> T >> B; pre(); while (T--) { int L, R; cin >> L >> R; cout << solve(R) - solve(L - 1) << '\n'; } cout.flush(); return 0; }
- 1
信息
- ID
- 3172
- 时间
- 1000ms
- 内存
- 1024MiB
- 难度
- 8
- 标签
- (无)
- 递交数
- 21
- 已通过
- 6
- 上传者