判断一个数是否为质数(素数)的4种方法

目录

1.什么是质数?

2.如何判断是否为质数?

方法1

方法2

方法3

方法4


1.什么是质数?

首先来看质数的概念:

质数(Prime number),又称素数,指在大于1的自然数中,除了1和该数自身外,无法被其他自然数整除的数。(也可定义为只有1与该数本身两个正因数的数)

 

图1  数字12不是质数,而数字11是质数

如上图所示,数字12可以将每4个分成一组,一共3组;而数字11将每4个、每5个、每3个分成一组都无法全部分完,而有剩余,因此将数字11称为质数。

 

2.如何判断是否为质数?

质数的特点如下:

一个自然数(如1、2、3、4、5、6等)若恰有两个正约数(1及此数本身),则称之为质数。

方法1

根据质数的约数只有1和本身这一特点,可以首先想到最直观的方法。第一种方法就是判断一个数是否能被比它小的数整除

方法1的时间复杂度是O(n)。

public static boolean isPrime(int n){
    //n<=3时,质数有2和3
    if (n <= 3) {
        return n > 1;
    }
    //当n>3时,质数无法被比它小的数整除
    for(int i = 2; i < n; i++){
        if (n % i == 0) {
            return false;
        }
    }
    return true;
}

方法2

当一个数不是质数时,必定存在两个约数,一个大于等于sqrt(n),另一个小于sqrt(n)。利用这种特性,可以对方法1进行改进,只判断数n能否被小于sqrt(n)的数整除。

方法2的时间复杂度是O(sqrt(n))。

图2  筛选判断集,只选择小于等于sqrt(n)的集合

 

public static boolean isPrime(int n) {
    if (n <= 3) {
        return n > 1;
    }
    //判断一个数能否被小于sqrt(n)的数整除
    int sqrt = (int)Math.sqrt(n);
    for (int i = 2; i <= sqrt; i++) {
        if(n % i == 0) {
            return false;
        }
    }
    return true;
}

方法3

任一偶数一定能分解为2和其他偶数/奇数的积,因此一个数不能被2整除,那么这个数一定不能被其他偶数整除。利用这个特点,可以对方法2进行改进,判断数n能否被小于sqrt(n)的奇数整除。

方法3的时间复杂度是O(sqrt(n)/2)。

图3  进一步筛选判断集,只选择小于等于sqrt(n)的奇数
public static boolean isPrime(int n) {
    if (n <= 3) {
        return n > 1;
    }
    //只需判断一个数能否被小于sqrt(n)的奇数整除
    int sqrt = (int)Math.sqrt(n);
    for (int i = 3; i <= sqrt; i += 2) {
        if(n % 2 == 0 || n % i == 0) {
            return false;
        }
    }
    return true;
}

方法4

质数的分布具有特点,经过证明可以得到,(大于等于5的)质数一定和6的倍数相邻,一定是6x-1或6x-1。利用这种特性。可以对整数进行筛选,只判断那些是6x-1或6x-1的整数是否为质数。

图4  筛选数据集,只选择6的倍数相邻的数

证明过程如下:

令x≥1,将大于等于5的自然数表示如下: ······6x-1,6x,6x+1,6x+2,6x+3,6x+4······(相邻6个数为一组)

在以上的数字中,6x、6x+2和6x+4是偶数,一定不是质数;6x+3可以分解为3(2x+1),不是质数,因此质数只能是6x-1和6x+1。

public static boolean isPrime(int n) {
    if (n <= 3) {
        return n > 1;
    }
    // 只有6x-1和6x+1的数才有可能是质数
    if (n % 6 != 1 && n % 6 != 5) {
        return false;
    }
    // 判断这些数能否被小于sqrt(n)的奇数整除
    int sqrt = (int) Math.sqrt(n);
    for (int i = 5; i <= sqrt; i += 6) {
        if (n % i == 0 || n % (i + 2) == 0) {
            return false;
        }
    }
    return true;
}

 

  • 341
    点赞
  • 982
    收藏
    觉得还不错? 一键收藏
  • 34
    评论
除了试除法,判断一个是否素数还有其他的方法,下面介绍几常用的方法。 1. 质数筛选法 质数筛选法是一较为常见的素数判断方法,其思想是先将待判断列出一些小于等于它的质数,然后判断是否能被这些质数整除。具体实现可以采用埃式筛法或欧拉筛法等,时间复杂度均为 O(nloglogn)。 2. Fermat测试 Fermat测试是一基于费马小定理的素数测试方法,其原理是:如果 p 是素数,且 a 是小于 p 的正整,则 a^(p-1) mod p = 1。因此,我们可以随机选取一些 a 值,进行计算,如果不等于 1,则 p 不是素数。如果等于 1,则 p 可能是素数,需要进行多次测试以提高准确性。 需要注意的是,Fermat测试对于一些合也可能返回为素数,因此需要多次测试以提高准确性。 3. Miller-Rabin算法 Miller-Rabin算法是一基于Fermat测试的素数测试算法,其原理是:对于一个 n,有至少一半的 a 值,满足 a^(n-1) mod n != 1。因此,我们可以随机选取一些 a 值,进行计算,如果不等于 1,则 n 不是素数。如果等于 1,则 n 可能是素数,需要进行多次测试以提高准确性。 需要注意的是,Miller-Rabin算法的准确性与测试次有关,测试次越多,准确性越高。一般来说,10次左右的测试已经可以满足绝大多情况。时间复杂度为 O(klog^3n),其中 k 为测试次。 以上是常见的几素数判断方法,每方法都有其优缺点,可以根据实际情况选择合适的方法

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值