数论-质数

质数

  • 质数基本概念
  • 质数的判定(试除法)
    • 定义判断
  • 分解质因数
  • 筛法求素数
    • 1.最普通的筛法 O(nlogn)
    • 2.诶氏筛法 O(nloglogn)粗略等于O(n)
    • 3.线性筛法 O(n)

质数基本概念

质数就是大于1的整数中只有1和本身两种因子的整数,或者叫素数

质数的判定(试除法)

定义判断

简单的根据定义进行判断即可,判断两个条件,第一是否大于1,第二是否满足只有1和本身两种因子
时间复杂度是O(n)的,由于因子是成对出现的,因此只需要遍历前半部分因子就行,也就是只需要遍历到sqrt(n),将时间复杂度降为O(根号n),数据量稍大也容易超时
(1)但是直接用sqrt(n)这个函数是比较慢的,不推荐,会容易超时,要么就先算出来,要么不用
(2)写作ii<=n,也不行,ii可能会爆int,
(3)写作i<=n/i比较稳妥
代码:

bool sushu(int a)
{
    if(a==1||a==0) return false;   //不满足第一个定义
    for(int i=2;i<=a/i;i++)     //到sqrt(a)就能遍历完所有因子了
    {
        if(a%i==0) return false;
    }
    return true;
}

分解质因数

对于任何一个正整数n,都可以分解成多个质数因子相乘,即p1,p2,p3…pk是质数因子,
n =p1a1 * p2a2 * p3a3 * …*pkak,pk作为质数底数,ak作为该质数的指数
刷题链接:https://www.acwing.com/problem/content/869/
求解思路:枚举所有素数因子,并且对每个因子循环求一下指数就行,
代码:

注意一个点,在枚举的时候不需要判断当前枚举到的因子是否是质数,因为当枚举到i时,说明a的2~i-1的因子已经都被除干净了,a中就不包含2 ~ i-1中的因子了,这时候如果i是a的因子,表示i中也不包含2 ~ i-1中的因子了(倍数都不包含,其中一个因子肯定也不包含啊),那么根据定义,当前i不包含2 ~ i-1中的任何因子,因此当前i就是一个质数,因此每次就不需要再多余判断了,不然用普通的定义法或者离线筛法都会超时
超时写法

#include<iostream>
#include<math.h>
#include<map>
using namespace std;
const int N=200005;
map<int,int>sushu;
//先给质数打个表
void init()
{
	//0表示是素数,1表示不是素数
	sushu[1]=1;sushu[0]=1; 
	sushu[2]=0;
	for(int i=2;i<=N;i++)
	{
		if(sushu[i]) continue;  //如果已经被标记为不是素数了,后面的倍数肯定也被标记了不是素数 
		for(int j=i*2;j<=N;j+=i)
		{
			if(!sushu[j]) sushu[j]=1;
		}
	} 
} 
int n,a;
int main()
{
	init();
	cin>>n;
	while(n--)
	{
		cin>>a;
		while(a>1)
		{
			for(int i=2;i<=a;i++)     
			{
				if((a%i==0)&&(sushu[i]==0))   //是因子,并且是素数 
				{
				    int t=0; 
				    while(a%i==0)  //求最多能除以多少个i 
				    {
				    	t++; a/=i;
				    }
				    cout<<i<<" "<<t<<endl;
				}
			}
		}
		cout<<endl;
	}
	return 0;
} 

AC代码:

#include<iostream>
#include<math.h>
using namespace std;
int n,a;
int main()
{
	cin>>n;
	while(n--)
	{
		cin>>a;
		for(int i=2;i<=a/i;i++)    //也是只需要枚举到根号a就行了
		{
			if(a%i==0)   //这个时候就已经能保证i是质数了,不需要多余一个判断 
			{
				int t=0; 
			    while(a%i==0)  //求最多能除以多少个i 
			    {
			    	t++; a/=i;
			    }
			    cout<<i<<" "<<t<<endl;
			}
		} 
		//a中最多只包含一个大于根号a的质因子,前面只枚举到根号a,如果最后剩下的值a大于1说明还存在这个大于根号a的质因子,单独输出,如果不存在,a除光了所有的因子,自己肯定是1了
		if(a>1) cout<<a<<' '<<1<<endl;
		cout<<endl;
	}
	return 0;
} 

注意一点,虽然定义法求素数和分解质因子的时间复杂度都是O(sqrt(n)),但是两者是不一样的,定义法一定会循环sqrt(n)词,但是分解质因子的a是不断地减小的,假如a是2^k次方,第一次除以k次2的时候出来a就变为1了,就结束了,因此分解质因子的时间复杂度严格来说应该是O(logn~sqrt(n)),比定义法的会快一点

筛法求素数

上面的直接遍历所有数,判断每个数是不是当前n的因子的时间复杂度O(n)还是有点高的,如果有m个数需要判断,时间复杂度就会是O(m*n),这种情况下有一种打表的标记素数的方式,筛法求素数提前把所需要的数据范围N内的素数全部标记一下,然后对m个数直接if判断就行,由于范围N内的每个数只会被标记一次,因此时间复杂度降为O(N+n)
但是对于每个数的取值范围N很大的情况也不友好,
比如这道题:https://www.acwing.com/problem/content/868/ ,数据范围达到了232,筛一遍肯定超时了
但是对于这道题:https://www.acwing.com/problem/content/870/,数据范围就是1e6,就能把O(n^2)时间复杂度降为O(1e6+n),达到优化的效果

筛法求素数也有多种求法,但是大致的思路都是先假定所有的数都是素数,然后再将那些不是素数的数筛出去,st[i]=true|1表示不是素数,st[i]=false|0表示是素数,初始时候让所有的元素的st都等于0,都是素数,然后再逐步地将不是素数的标记为1

1.最普通的筛法 O(nlogn)

从前往后遍历数据范围N内的所有数,每次遍历都把当前数的倍数都给筛出去。
简单证明:假如当前要筛选的数为p,如果2 ~ p-1的数没能把p筛出去,就说明p不是2~p-1中任意一个数的倍数,也就是是回归定义,p在2 ~ p-1不存在因子 ,因此p是质数
时间复杂度:n/2+n/3+…n/n =nln(n)=nlogn

void get_primes2(){
    for(int i=2;i<=n;i++){
        if(!st[i]) primes[cnt++]=i;//把素数存起来
        for(int j=i;j<=n;j+=i){//不管是合数还是质数,都用来筛掉后面它的倍数
            st[j]=true;
        }
    }
}

2.诶氏筛法 O(nloglogn)粗略等于O(n)

只需要用质数来将它的倍数都筛掉,因为如果对合数也筛掉它的倍数的话,实际在前面他的因子已经将这个合数以及这个合数的所有倍数都筛出去了,不需要再筛一遍了,重复。能比普通筛法快3倍左右

void get_primes1(){
    for(int i=2;i<=n;i++){
        if(!st[i]){
            primes[cnt++]=i;
            for(int j=i;j<=n;j+=i) st[j]=true;//可以用质数就把所有的合数都筛掉;
        }
    }
}

3.线性筛法 O(n)

当数据范围达到1e7的情况下,线性筛法的速度能达到埃氏筛法的一倍左右
前面的两种筛法对每个合数进行筛除的时候可能存在重复筛除的情况,比如在埃氏筛法中15就会被3和5两个质数进行筛除,在普通筛法中每个数还会被合数因子筛除。
线性筛法就保证了每个数只会被自己的最小质因子筛除,不存在重复筛除的情况,会减少时间复杂度
代码:

void get_primes()
{
	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]数组下标不越界,也就是只需要标记到第n个数
		{   //不需要加j<cnt的条件,因为循环一定会在if(i%primes[j]==0) break;的时候跳出,因为如果i是合数,一定会存在最小质因子使条件满足,如果i是质数,在枚举到primes[j]=i的时候也就跳出了
			st[primes[j]*i]=true;  //标记为不是质数
			if(i%primes[j]==0) break;    //这时候primes[j]一定是i的最小质因子,因为是从前往后遍历所有质数,到达一个因子就跳出 
		}
	}
}

先记住模板,然后我们再来证明为什么每个合数只会被他的最小质因子筛除,并且还要证明所有的合数都会被筛除,最后剩下的就全都是质数了。
证明一:每个合数只会被他的最小质因子筛除
在循环里的 st[primes[j]*i]=true; 进行筛除的时候,存在两种情况
第一种情况:i%primes[j] == 0 , 由于是从小到大遍历所有的质数,因此表示primes[j]是i的最小质因子,也就说明primes[j]是primes[j]*i的最小质因子
第二种情况:i%primes[j] != 0的时候 ,表示primes[j]小于i的最小质因子,也就说明primes[j]是primes[j]*i的最小质因子
得证,每次筛除primes[j]*i(合数)的时候都是用其最小质因子筛除的

证明二:所有合数都会被筛除
易知任何一个约数x,都会在循环遍历到它之前被筛出,因为在遍历到它之前,总会出现primes[j]*i=x(每个合数都存在一个最小质因子)

因此从上可以看出,由于每个数只有一个最小质因子,每个数只会被自己的最小质因子筛除,因此每个数只会被筛一次,时间复杂度保证了是O(n)线性的

革命尚未成功,同志仍需努力…

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/770.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

链表相关oj题

1.Leetcode203 移除链表元素 解题思路&#xff1a;从头节点开始进行元素删除&#xff0c;每删除一个元素&#xff0c;需要重新链接节点 struct ListNode* removeElements(struct ListNode* head, int val){struct ListNode*dummyheadmalloc(sizeof(struct ListNode));dummyhea…

spring5(四):IOC 操作 Bean 管理(基于注解方式)

IOC操作Bean管理&#xff08;基于xml方式&#xff09;前言一、注解1、概述二、入门案例1、Bean 的创建2、Bean的自动装配2.1 Autowired2、Qualifie3、Resource4、Value3、扫描组件3.1 配置文件版3.2 注解版4、测试前言 本博主将用CSDN记录软件开发求学之路上亲身所得与所学的心…

Mysql常用命令

mysql连接&#xff1a; [roothost]# mysql -u root -p Enter password:******创建数据库&#xff1a; CREATE DATABASE 数据库名&#xff1b; 删除数据库&#xff1a; drop database 数据库名; 使用mysqladmin删除数据库&#xff1a; [roothost]# mysqladmin -u root -p dr…

【数据结构】链表OJ(二)

Yan-英杰的博客 悟已往之不谏 知来者之可追 目录 一、反转链表 二、合并两个有序链表 三、链表分割 四、链表的回文结构 一、反转链表 输入&#xff1a;head [1,2,3,4,5] 输出&#xff1a;[5,4,3,2,1] 输入&#xff1a;head [1,2] 输出&#xff1a;[2,1] 示例 3&#xf…

Vulnhub靶场----10、LazySysadmin

文章目录一、环境搭建二、渗透流程一、环境搭建 DC-7下载地址&#xff1a;https://download.vulnhub.com/dc/DC-9.zip kali&#xff1a;192.168.144.148 DC-9&#xff1a;192.168.144.157 二、渗透流程 1、信息收集nmap -sV -sT -p- -T4 192.168.144.157思路&#xff1a; 1、80…

基于vivado(语言Verilog)的FPGA学习(3)——FPGA理论知识

基于vivado&#xff08;语言Verilog&#xff09;的FPGA学习&#xff08;3&#xff09;——FPGA理论知识 文章目录基于vivado&#xff08;语言Verilog&#xff09;的FPGA学习&#xff08;3&#xff09;——FPGA理论知识1. FPGA介绍1.1.FPGA内部结构&#xff08;1&#xff09;. 可…

【云原生|Docker】01-docker简介

目录 前言 Docker简介 1. 什么是docker 2. Docker和vm有什么区别 3. Docker架构 4. Docker特性 Docker安装 1. Docker版本介绍 2. Centos7安装docker 3. Docker校验 4. Docker启动 5. Docker配置文件 前言 接下来准备记录云原生系列的相关知识&#x…

Linux防火墙的关闭

查看防火墙的状态打开终端输入如下命令systemctl status firewalld如图所示&#xff1a;running表示防火墙目前处于打开状态输入命令进行关闭防火墙&#xff1a;systemctl stop firewalld如图所示正常的用户是没有权限的&#xff0c;需要输入管理员的密码才能够进行关闭防火墙。…

OpenAI GPT-4震撼发布:多模态大模型

OpenAI GPT-4震撼发布&#xff1a;多模态大模型发布要点GPT4的新功能GPT-4:我能玩梗图GPT4:理解图片GPT4:识别与解析图片内容怎样面对GPT4申请 GPT-4 API前言&#xff1a; &#x1f3e0;个人主页&#xff1a;以山河作礼。 &#x1f4dd;​&#x1f4dd;:本文章是帮助大家更加了…

中国版的“ChatGPT”狂飙的机会或许要出现了

⭐️我叫忆_恒心&#xff0c;一名喜欢书写博客的在读研究生&#x1f468;‍&#x1f393;。 如果觉得本文能帮到您&#xff0c;麻烦点个赞&#x1f44d;呗&#xff01; 近期会不断在专栏里进行更新讲解博客~~~ 有什么问题的小伙伴 欢迎留言提问欧&#xff0c;喜欢的小伙伴给个三…

avue-crud组件的行内编辑实现失焦保存,在没有右侧操作栏的情况下

前言 关于 avue 框架&#xff0c;其实本来不想写一篇随笔记录的&#xff0c;因为目前在网上有很多文章&#xff0c;关于其配置项介绍的比较详细&#xff0c;而且官网上也有对应的文档&#xff0c;这两者结合足以满足大部分的开发需求。 不过&#xff0c;产品经理总会有些不一…

[大二下]什么是NPM

[大二下]什么是npm? 什么是NPM? 最简单来回答: ​ 就是一个包管理器, 一个仓库, 谁需要里面的物品, 谁就拿 npm 全称 Node Package(译: 包,包裹) Manager(译:如下). 直译过来就是 Node的包管理, 但是我们真正咱们约定俗成的称 NPM为"Node的包管理器". npm是Jav…

nvm使用-node版本切换-npm版本-node版本异常导致错误

目录什么是nvm?为什么要用它&#xff1f;它改变的是谁的版本号&#xff1f;安装并使用安装前操作安装使用&#xff08;常用命令&#xff09;nvm -hnvm install \<version\> [arch]nvm listnvm use [version] [arch]其他什么是nvm? .nvm是一个node的版本管理工具&#x…

【计算机图形学】扫面转换算法(DDA算法 中点画线算法 Bresenham画线算法)

模块1 扫描转换算法 一 实验目的 编写直线、弧线的光栅扫描转换算法&#xff0c;并对线宽与线形的算法加以探讨用DDA算法、中点画线算法、Bresenham画线算法绘制直线&#xff08;如果键盘输入数据&#xff0c;给出数据值&#xff1b;如果绘制图案&#xff0c;图案中应包含各种…

机器看世界

博主简介 博主是一名大二学生&#xff0c;主攻人工智能研究。感谢让我们在CSDN相遇&#xff0c;博主致力于在这里分享关于人工智能&#xff0c;c&#xff0c;Python&#xff0c;爬虫等方面知识的分享。 如果有需要的小伙伴可以关注博主&#xff0c;博主会继续更新的&#xff0c…

开源超级终端工具——WindTerm

1、下载和安装&#xff08;我的是win10&#xff0c;其他版本各位自选&#xff09; Releases kingToolbox/WindTerm GitHub 安装的话&#xff0c;相信大家不用我赘述了。 初始界面是这样的&#xff1a; 2、WindTerm使用 2.1 本地会话&#xff08;最下面那个框&#xff0c;发…

自动化测试实战篇(10),找不到合适接口测试怎么办?Postman中mock模拟接口帮你解决烦恼

一般想学习接口测试&#xff0c;找不到相应的接口进行测试也是比较麻烦的一件事情&#xff0c;尤其是找一些能够正常显示想要的相应的数据的接口更是相对来讲比较复杂&#xff0c;那么有没有简单点造接口数据的方式呢&#xff1f; 像是mock框架&#xff0c;以它为基础的apifox…

23.3.14打卡 2022年江西省大学生程序设计竞赛(正式赛)ABL

就写了签到, 其他题没写, 这场好像3题就银了 纪念一下3.14原粥率日 比赛链接:https://ac.nowcoder.com/acm/contest/43898 A题 Special Adjustment Method 题意 给出非负整数x, y, z 你可以让其中两个数字-1, 另外一个2, 使得x2y2z2x^2y^{2}z^{2}x2y2z2最大 题解 这题很容…

站上风口,文心一言任重道远

目录正式发布时机选择逻辑推理AI绘画用户选择总结自从OpenAI公司的chatGPT发布以来&#xff0c;吸引了全球目光&#xff0c;同时也引起了我们的羡慕&#xff0c;希望有国产的聊天机器人&#xff0c;盼星星盼月亮&#xff0c;终于等来了百度文心一言的发布。 正式发布 3月16日…

安全SaaS,在中国TO B中艰难成长

无论是一体化、还是以业务为中心专攻政企或金融客户&#xff0c;还是针对中小微企业市场推出免费产品&#xff0c;都可能成为未来安全SaaS规模化的发展路径。 作者|斗斗 编辑|皮爷 出品|产业家 5G、物联网、AI、云计算等技术的应用&#xff0c;让生产、服务过程加速数字化、…
最新文章