质数相关知识

判断质数

对于一个数n,不借助后面质数数组的情况下,我们可以从2往后遍历来判断n是否有因子,从而判断其是否为质数。也就是从小到大找第一个质因子。时间复杂度O()。

cpp 复制代码
bool is_prime(int n)
{
    for(int i = 2; i <= n  / i; i ++)
        if(n % i == 0)return false;

    return true;
}

筛质数

目标:获得质数数组、O(1)判断一个数是不是质数

扩展:获得一个数的最小质因子

所需数组:

cpp 复制代码
int primes[N], cnt; // 质数数组
bool st[N]; // 判断质数
int p[N]; // 最小质因子(可选)

普通筛,用遍历到的数的倍数筛掉合数,时间复杂度O(nloglog n)

cpp 复制代码
void sieve(int n)
{
    for(int i = 2; i <= n; i ++)
    {
        if(!st[i])primes[cnt ++] = i;
        for(int j = i * 2; j <= n; j += i)
            st[j] = true;
    }
}

可以发现,在遍历的i时候明显重复给某些合数赋值了,比如12会在i=2,i=3,i=4,i=6的时候反复被筛掉。

埃式筛,用质数的倍数去筛掉合数,时间复杂度O(nloglog n)

cpp 复制代码
void sieve(int n)
{
    for(int i = 2; i <= n; i ++)
    {
        if(!st[i])
        {
            primes[cnt ++] = i;
            for(int j = i * 2; j <= n; j += i)
                st[j] = true;
        }
    }
}

在遍历的i时候还是会有某些质数重复给某些合数赋值,比如12会在i=2,i=3的时候反复被筛掉。

线性筛,用最小的质因子去筛掉对应合数,时间复杂度O(n)

cpp 复制代码
void sieve(int n)
{
    for(int i = 2; i <= n; i ++)
    {
        if(!st[i])primes[cnt ++] = i;
        for(int j = 0; primes[j] <= n / i; j ++)
        {
            st[primes[j] * i] = true;
            if(i % primes[j] == 0)break;
        }
    }
}

可以分析一下为什么是这样的:

因为j从小到大遍历,即质数从小到大遍历,所以primes[j]一定小于等于i的最小质因子,因为

if(i % primes[j] != 0),那么说明i的最小质因子还没有被遍历到,

if(i % primes[j] == 0),这时候说明i的最小质因子就是primes[j],直接break,

这样就可以保证primes[j] * i的最小质因子就是primes[j]

所以我们就一直在用最小质因子去筛合数,每个合数只会被遍历一次。

存下这个最小质因子就很简单了

cpp 复制代码
void sieve(int n)
{
    for(int i = 2; i <= n; i ++)
    {
        if(!st[i])primes[cnt ++] = i, p[i] = i;
        for(int j = 0; primes[j] <= n / i; j ++)
        {
            st[primes[j] * i] = true;
            p[primes[j] * i]  = primes[j];
            if(i % primes[j] == 0)break;
        }
    }
}

分解质因数

一个大于1的数可以被分解为若干个质数的乘积

从小到大找即可,下面输出质因子及个数

cpp 复制代码
void work(int x)
{
    for(int i = 2; i <= x / i; i ++)
    {
        int cnt = 0;
        if(x % i == 0)
        {
            while(x % i == 0)
            {
                x /= i; 
                cnt ++;
            }
            cout<<i<<' '<<cnt<<endl;
        }
    }
    if(x > 1)cout<<x<<' '<<1<<endl;
}
相关推荐
pkowner4 分钟前
若依分页问题及解决方法
java·前端·算法
呃呃本27 分钟前
算法题(栈)
算法
通信小呆呆34 分钟前
基于 ADMM-MFOCUSS 的捷变频雷达扩展目标稀疏重构原理
算法·重构·信息与通信·信号处理·雷达
橙淮44 分钟前
Java数组与链表:特性对比与应用场景
数据结构·算法
炽烈小老头1 小时前
【每天学习一点算法 2026/05/15】被围绕的区域
学习·算法·深度优先
芜湖xin1 小时前
【题解-洛谷】P1012 [NOIP 1998 提高组] 拼数
算法·贪心
xiaoxiaoxiaolll2 小时前
金属结构疲劳寿命预测与健康监测技术
人工智能·算法·机器学习
故事和你912 小时前
洛谷-【图论2-1】树4
开发语言·数据结构·c++·算法·动态规划·图论
故事和你912 小时前
洛谷-【图论2-1】树1
开发语言·数据结构·c++·算法·深度优先·动态规划·图论
敲代码的嘎仔2 小时前
力扣高频SQL基础50题详解
开发语言·数据库·笔记·sql·算法·leetcode·后端开发