求约数和的三重境界
求约数和的三重境界
一、先上结论
数据量/办法 | 暴力 | 普通筛法 | 欧拉筛法 |
---|---|---|---|
无法忍受,不能出结果 |
考虑到的复杂度,按一秒的运算能力,,应该是需要秒,三个小时,去死吧!太垃圾了~
注意:
- 普通筛法:代码极短,性能足够,推荐使用
- 欧拉筛法:代码太长,性能优秀,不推荐使用
二、暴力法
//暴力大法 , 不包含自己
int bforce(int n) {
int sum = 0;
for (int i = 1; i < n; i++) {
if (n % i == 0)
sum = sum + i;
}
return sum;
}
时间复杂度
三、普通筛法
for (int i = 1; i <= n; i++) // O(n)
for (int j = 2; j <= n / i; j++) // 调和级数O(logn) = n + n / 2 + n / 3 + ... + n/n = lnn + c
sum2[i * j] += i;
时间复杂度
四、线性筛法
int primes[N], cnt; // primes[]存储所有素数
bool st[N]; // st[x]存储x是否被筛掉
int num[N]; // i的最小质因子构成的一个等比数列求和的结果
int sd[N]; //约数和
void get_primes(int n) {
sd[1] = 1; // 1的约数和是1
for (int i = 2; i <= n; i++) { //遍历 2~n,筛质数的同时,筛出每个数字的约数和
if (!st[i]) { //如果i是质数
primes[cnt++] = i; //记录i到质数数组中
sd[i] = num[i] = 1 + i; //质数的约数和是1+自身
}
for (int j = 0; primes[j] * i <= n; j++) {
st[i * primes[j]] = true; //记录primes[j]的整数倍(i倍)是合数
if (i % primes[j]) { //如果i不包含primes[j]这个质数因子
sd[i * primes[j]] = sd[i] * sd[primes[j]];
num[i * primes[j]] = primes[j] + 1;
} else { //如果i包含了primes[j]这个质数因子
sd[i * primes[j]] = sd[i] / num[i] * (num[i] * primes[j] + 1);
num[i * primes[j]] = num[i] * primes[j] + 1;
break;
}
}
}
}
五、附:测试对比的代码
#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
const int N = 1e6 + 10;
int sum1[N], sum2[N], sd[N];
int n = 1e5;
//暴力大法 , 不包含自己
int bforce(int n) {
int sum = 0;
for (int i = 1; i < n; i++) {
if (n % i == 0)
sum = sum + i;
}
return sum;
}
int primes[N], cnt; // primes[]存储所有素数
bool st[N]; // st[x]存储x是否被筛掉
int num[N]; // i的最小质因子构成的一个等比数列求和的结果
int sd[N]; //约数和
void get_primes(int n) {
sd[1] = 1; // 1的约数和是1
for (int i = 2; i <= n; i++) { //遍历 2~n,筛质数的同时,筛出每个数字的约数和
if (!st[i]) { //如果i是质数
primes[cnt++] = i; //记录i到质数数组中
sd[i] = num[i] = 1 + i; //质数的约数和是1+自身
}
for (int j = 0; primes[j] * i <= n; j++) {
st[i * primes[j]] = true; //记录primes[j]的整数倍(i倍)是合数
if (i % primes[j]) { //如果i不包含primes[j]这个质数因子
sd[i * primes[j]] = sd[i] * sd[primes[j]];
num[i * primes[j]] = primes[j] + 1;
} else { //如果i包含了primes[j]这个质数因子
sd[i * primes[j]] = sd[i] / num[i] * (num[i] * primes[j] + 1);
num[i * primes[j]] = num[i] * primes[j] + 1;
break;
}
}
}
}
int main() {
//记录开始时间
clock_t begin = clock();
// 1、暴力计算 O(n^2)
// for (int i = 1; i <= n; i++) sum1[i] = bforce(i);
// for (int i = 1; i <= n; i++) printf("%d ", sum1[i]);
puts("");
//记录结束时间
clock_t end = clock();
cout << "Running time:" << (double)(end - begin) / CLOCKS_PER_SEC * 1000 << "ms" << endl;
// 2、通过筛法求出1到n的所有约数之和
//记录开始时间
begin = clock();
for (int i = 1; i <= n; i++) // O(n)
for (int j = 2; j <= n / i; j++) // 调和级数O(logn) = n + n / 2 + n / 3 + ... + n/n = lnn + c
sum2[i * j] += i;
// for (int i = 1; i <= n; i++) printf("%d ", sum2[i]);
puts("");
//记录结束时间
end = clock();
cout << "Running time:" << (double)(end - begin) / CLOCKS_PER_SEC * 1000 << "ms" << endl;
// 3、利用欧拉定理筛出约数和
//记录开始时间
begin = clock();
get_primes(n);
// for (int i = 1; i <= n; i++) printf("%d ", sd[i]-i);
puts("");
//记录结束时间
end = clock();
cout << "Running time:" << (double)(end - begin) / CLOCKS_PER_SEC * 1000 << "ms" << endl;
return 0;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 【自荐】一款简洁、开源的在线白板工具 Drawnix
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· Docker 太简单,K8s 太复杂?w7panel 让容器管理更轻松!
2021-09-02 等比数列求和公式
2017-09-02 访问Github慢的解决办法
2017-09-02 CentOS6.9下安装 Pika 2.2.5(新增了拷贝安装版本的办法+对于PID的位置及数据库位置的理解)
2016-09-02 鞋业管理系统定期执行任务
2016-09-02 针对各地项目icomet停止服务的临时处理办法