快速高效求素数|质数的方法—Java(模板)

判断素数|质数方法时间效率:线性筛法>埃氏筛法>试除法

在写算法题的时候,各种各样跟素数有关的题目非常常见,本文列出了三种常见的判断素数的方法

三种求素数方法的优缺点

一、试除法

试除法的基本思想是:判断一个数 x 是否为素数,需要检查从 2 到 sqrt(x)(即 x/i)之间的所有数。如果存在一个数 i 能整除 x,则 x 不是素数。我们只需要检查到 sqrt(x),因为如果 x = a * b,那么必定存在一个 a <= sqrt(x) 和一个 b >= sqrt(x),因此我们只需要检查较小的因子。

注意:sqrt()方法的计算速度较慢,并且为了防止溢出的问题,建议写成i<=x/i

java 复制代码
 static boolean get(int x){
      if(x<2)return false;
      for(int i=2;i<=x/i;i++){
          if(x%i==0)return false;
      }
      return true;
 }

二、埃氏筛法

埃氏筛法:

假设所有数都为素数,从 2 开始。

对于每个素数 i,将其所有的倍数标记为合数(即不是素数)。

继续处理下一个未标记的数作为素数,直到遍历到 x。

最终,未被标记的数即为素数。

数据量大于10e7可能会超时

java 复制代码
    static void get(int x){
        for(int i=2;i<=x;i++){
            if(!flag[i]){
                pri[cnt++]=i;
                for(int j=i+i;j<=x;j+=i)flag[j]=true;
            }
        }
    }

三、线性筛法

线性筛法是在埃氏筛法的基础上进行优化。

区别:

埃氏筛法:每次遍历素数时都需要标记所有倍数为合数。

线性筛法:只标记 i 与当前已有的素数相乘的倍数(不再处理重复倍数)。这样可以避免多次重复标记,提高效率。

筛法过程:

遍历每个数 i,如果 i 是素数(即 flag[i] == false),则将其加入素数列表。

对于每个素数 p,从 i * p 开始标记合数,直到 x。

注意标记时,若 i 已经被 p(一个较小的素数)整除,则不再继续与更大的素数进行标记,避免重复。

java 复制代码
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.StreamTokenizer;
import java.util.ArrayList;
import java.util.List;

public class Main {
    static int N=1000010;
    static List<Integer> list=new ArrayList<>();
    static boolean flag[]=new boolean[N];
    public static void main(String[] args) throws IOException {
        Read sc=new Read();
        int n=sc.nextInt();
        get(n);
        System.out.println(list.size());
    }
    static void get(int n){
        for(int i=2;i<=n;i++){
            if(!flag[i])list.add(i);
            for(int j=0;list.get(j)<=n/i;j++){
                flag[list.get(j)*i]=true;
                if(i%list.get(j)==0)break;
            }
        }
    }
}

class Read{
    StreamTokenizer st=new StreamTokenizer(new BufferedReader(new InputStreamReader(System.in)));
    public int nextInt() throws IOException {
        st.nextToken();
        return (int)st.nval;
    }
}
相关推荐
=>>漫反射=>>1 分钟前
单元测试 vs Main方法调试:何时使用哪种方式?
java·spring boot·单元测试
初圣魔门首席弟子2 分钟前
c++ bug 记录(merge函数调用时错误地传入了vector对象而非迭代器。)
java·c++·bug
ZhengEnCi12 分钟前
@Parameter 注解技术解析-从 API 文档生成到接口描述清晰的 SpringBoot 利器
java·spring boot
Swift社区13 分钟前
从 0 到 1 构建一个完整的 AGUI 前端项目的流程在 ESP32 上运行
前端·算法·职场和发展
低调小一14 分钟前
Kuikly 小白拆解系列 · 第1篇|两棵树直调(Kotlin 构建与原生承载)
android·开发语言·kotlin
RTC老炮17 分钟前
webrtc弱网-BitrateEstimator类源码分析与算法原理
网络·人工智能·算法·机器学习·webrtc
郝学胜-神的一滴18 分钟前
Linux下的阻塞与非阻塞模式详解
linux·服务器·开发语言·c++·程序人生·软件工程
AresXue20 分钟前
2025最新Java性能优化建议 应用 数据库 机器 网络
java
程序员烧烤21 分钟前
【leetcode刷题007】leetcode116、117
算法·leetcode
跟着珅聪学java30 分钟前
spring boot 整合 activiti 教程
android·java·spring