朝题夕解之数论

🌟分解质因数

算术基本定理: 对于任何一个大于1的正整数都能唯一分解为有限个质数的乘积,可写作:

N = P1c1 P2c2 …Pmcm

其中ci都是正整数,Pi都是质数,且满足P1 < P2 < … < Pm

🌻例题描述

分解质因数
🎇🎇🎇原题传送门🎇🎇🎇

🌻参考代码(C++版本)
#include <iostream>
#include <algorithm>

using namespace std;

void divide(int n)
{
    //试除法枚举所有的数
    for(int i = 2;i <= n / i;i++)
        if(n % i == 0)
        {
            int s = 0;
            while(n % i == 0)
            {
                n /= i;
                s++;
            }
            printf("%d %d\n",i,s); //要清楚i才是底数
        }
        
    if(n > 1) printf("%d %d\n",n,1);
    puts("");
}


int main()
{
    //输入
    int n;
    scanf("%d",&n);
    while(n--)
    {
        //调函数
        int x;
        scanf("%d",&x);
        divide(x);
    }
    return 0;
}
🌻算法模板

一、算法实现流程图:
分解质因数实现流程
二、算法的代码实现:

试除法分解质因数

void divide(int x)
{
    for (int i = 2; i <= x / i; i ++ )
        if (x % i == 0)
        {
            int s = 0;
            while (x % i == 0) x /= i, s ++ ;
            cout << i << ' ' << s << endl;
        }
    if (x > 1) cout << x << ' ' << 1 << endl;
    cout << endl;
}
🌻疑难点剖析

一、明白题目需求
需求

题目要的是将传入的数据分解为底数是/质数,指数是整数,分解结果的乘积等于原数据,输出的时候,要从小到大排列。

例如:
6就是21 x 31,因此就输出 2 1和3 1
48就是24 x 31,因此就输出2 4 和 3 1

二、指数的获取
8 = 23 = 2 x 2 x 2;

那么在代码层面,我们可以采用逆向思维,倒着对8进行运算,统计它进行的除法次数,也就是相乘的次数了。同时,也就是指数了。

🌟质数筛选

🌻例题描述

质数筛选
🎇🎇🎇原题传送门🎇🎇🎇

参考代码(C++版本)
// 埃氏筛选法
#include <iostream>
#include <algorithm>

using namespace std;

const int N = 1e6 + 10;
int primes[N],cnt;
bool st[N];


void get_primes(int n)
{
    for(int i = 2;i <= n;i++)
    {
        if(!st[i])
        {
            primes[cnt++] = n;
            //利用现有的质数,将质数的倍数去掉
            for(int j = i+i; j <= n;j += i) st[j] = true;
        }
    }
}

int main()
{
    //输入
    int n;
    cin >>n;
    //调用函数
    get_primes(n);
    //输出
    cout << cnt <<endl;
    
    return 0;
}


//线性筛选法  算法原理:n只会被最小质因子筛掉
#include <iostream>
#include <algorithm>

using namespace std;

const int N = 1e6 + 10;
int primes[N],cnt;
bool st[N];


void get_primes(int n)
{
    for(int i = 2;i <= n;i++)
    {
        //如果是质数,就把加到数组中去
        if(!st[i]) primes[cnt ++]  = i;
        //从小到大枚举所有的质数
        for(int j = 0; primes[j] <= n/i;j++)
        {
            //把prims[j] * i筛掉
            st[primes[j] * i] = true;
            if(i % primes[j] == 0) break; //primes[j]一定是i的最小质因子
        }
        
        /*
        1、i % pj == 0 
            pj一定是i的最小质因子,pj一定是pj*i的最小质因子
        2、i % pj != 0
            pj一定小于i的所有质因子,pj也一定是pj * i的最小质因子
        */
    }
}

int main()
{
    //输入
    int n;
    cin >>n;
    //调用函数
    get_primes(n);
    //输出
    cout << cnt <<endl;
    return 0;
}
🌻算法模板

一、埃氏筛法——用当前已有的质数去消去它们的倍数

首先,将2到n范围的所有整数获取到。其中,最小的数字2是质数,将2的所有倍数划去。表中剩余的数字中,最小的是3,它不能被更小的整除,所有它是质数,再将所有3的倍数划去。以此类推如果表中剩余的最小数字是m时,m就是质数。然后将表中所有m的倍数都划去。像这种反复操作,就能够依次枚举n以内的质数。

举个栗子:
埃氏算法例子

1.1、 算法实现流程如下:
埃氏筛除法

1.2、算法代码实现:

埃氏筛法求素数
int primes[N], cnt;		// primes[]存储所有素数
bool st[N];			// st[x]存储x是否被筛掉

void get_primes(int n)
{
    for (int i = 2; i <= n; i ++ )
    {
        if (st[i]) continue;
        primes[cnt ++ ] = i;
        for (int j = i; j <= n; j += i)
            st[j] = true;
    }
}


1.3、算法的时间复杂度 = O(NloglogN)
二、线性筛法

线性筛选的核心是—— 传入的整数n只会被最小质因子筛掉

2.1、算法实现流程:
线性筛法
2.2、算法代码实现:

线性筛法求素数
int primes[N], cnt;		// primes[]存储所有素数
bool st[N];			// st[x]存储x是否被筛掉

void get_primes(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;
        }
    }
}

2.3、算法时间复杂度 = O(N)

🌻疑难点剖析
清楚primes数组和st数组是分别用来维护素数集合和被筛除数据的集合
  • 6
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

杨枝

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值