求素数的三大算法 —— Java 篇

求素数的三大算法 —— Java 篇


  • 素数又叫质数(prime number),有无限个。质数定义为在大于1的自然数中,除了1和它本身以外不再有其他因数。

算法一 :暴力遍历

思路:

  1. 素数我们运用两个循环的遍历的方式,首先根据素数的定义:质数定义为在大于1的自然数中,除了1和它本身以外不再被其数整除了。因为任何数都可以被 1 整除的,所以我们的第一个for( )循环语句的初始值从 2 开始进行,到你所需的范围。
  2. 我们的第二个for()循环语句,的判断条件是:从上个for()循环语句的 变化值开始,但是注意不要等于 因为不要除以它本身,其中我们 if() 语句判断是否有,被除了它本身以外的数整除,同时我们定义一个变量,用于标记该数是否为被整除,逆向思考,被整除了,就不是素数,否则为素数
  3. 最后我们通过我们的标记,判断素数需要打印数值,
  4. 查找的时间复杂度为 O(n)

代码:


public class Study13 {
    /* 求素数的三大算法*/
    // 算法 —— : 暴力遍历
    public static void main (String[] args) {
        Scanner scanner = new Scanner (System.in);
        if(scanner.hasNextInt()) {
            int n = scanner.nextInt();
            int count = 0;
            for (int i = 2; i <= n; i++) {
                int sign = 1; // 用于标记是否是素数
                for (int j = 2; j < i; j++) {
                    if (i % j == 0 ) {
                        sign = 0;
                        break;
                    }
                }
                if (sign == 1) {
                    System.out.print(i+"\t");
                    count++;
                    if(count % 5 == 0) {
                        System.out.println();
                    } // 2,3,5,7,11,13,17,19,23,29,31,37,41,43,53,59,61,67,71,73,79,83,89,97。
                }
            }
        }
        scanner.close(); // 关闭 IO 流
    }
}

结果:

在这里插入图片描述


算法二:折半范围遍历

思路:

  • 在第一个算法的基础上,改进之法
  • 请看下图所示:

在这里插入图片描述

  • 我们仔细观察上述的图解:发现其规律没有:
    • 所有的数的可以被整除的最大的,除数都是 {其数值的 / 2 的 }如:18 的可以被最大的整除的是 (18/2 =) 9 , 12 可以被最大的整除的数是 (12 /2 =) 6 ;
    • 所以我们只要判断其数值的 1/2的范围内 是否有被整除 (除 1 外) 就可以了,会被整除就不是素数,不会整除就是素数
  • 逆向思考,这样我们就缩小了遍历范围,在第一种算法的基础上减半,
  • 查找的时间复杂度为 O(n/2)

代码:

import java.util.Scanner;
import java.util.concurrent.ForkJoinPool;

public class Study13 {
    /* 求素数的三大算法*/

    // 算法 二: 折半遍历
    public static void main (String[] args) {
        Scanner scanner = new Scanner (System.in);
        int count = 0;
        if(scanner.hasNextInt()) {
            int n = scanner.nextInt();
            for (int i = 2; i <= n; i++) {
                int sign = 1;
                for (int j = 2; j <= i/2 ; j++) {
                    if(i % j == 0) {
                        sign = 0;
                        break;
                    }
                }
                if(sign == 1) {
                    System.out.print(i+"\t");
                    count++;
                    if(count % 5 == 0) {
                        System.out.println();
                    }
                }

            }
        }
        scanner.close(); // 关闭 IO流
    }
}

结果:

在这里插入图片描述


算法三:根号范围遍历

思路:

  • 同样是在第一个算法的基础上改进的

  • 请看下面的图示:

    • 在这里插入图片描述

  • 仔细观察图示,有没有发现,我们求他们的平方根看看,结果:

  • 发现在它们的平方范围内,都会有可以被整除的数值,(当然我们这里不是素数的数值)逆向思考:不是的不就是素数了吗?

  • 这样我们再次缩小了遍历的范围了,在第一种算法的基础上减了,一个根号范围;

  • 查找的时间复杂度为 O(根号n )


代码:

import java.util.Scanner;


// 算法三.根号范围遍历
public class Study13 {

    public static void main (String[] args) {
        Scanner scanner = new Scanner (System.in);
        int count = 0;
        if(scanner.hasNextInt()) {
            int n = scanner.nextInt();
            for (int i = 2; i <= n; i++) {
                int sign = 1;
                for (int j = 2; j <= (int)Math.sqrt(i) ; j++) { 
                    if(i % j == 0) {
                        sign = 0;
                        break;
                    }

                }
                if(sign == 1) {
                    System.out.print(i+"\t");
                    count++;
                    if(count % 5 == 0) {
                        System.out.println("");
                    }
                }

            }
        }

        scanner.close(); // 关闭IO 流

    }
}

结果:

在这里插入图片描述


总结:

  • 总的来说通过找寻到某种规律:从而缩小查找的范围:如第一种算法:中我们查找的范围是 n , 第二种算法中:我们查找的范围是 n / 2 ,第三中算法:我们的查找范围是 根号 n
  • 从而达到实现提高程序的效率

最后:

每博一文案

你看,我早知道有很多的道理,我们并不是不知道,可人生的某些事总要自己经历过,才能真正把生活的意义都明了,没有谁可以复制谁的人生硕果,也没有谁可以规避谁的人生坎坷一句我早知道于事无补于心,有格。试着去拥抱那些满身伤口的人吧,我有陪伴和安慰,才能让那些伤口在开出花朵。

​ ———— 一禅心灵庙语

限于自身水平有限,其中存在的错误希望大家给予指教,谢谢大家,韩信点兵——多多益善!,后会有期,江湖再见!

  • 13
    点赞
  • 63
    收藏
    觉得还不错? 一键收藏
  • 4
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值