百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术资源 > 正文

2020-09-20:如何判断一个数是质数?

off999 2024-12-11 16:24 20 浏览 0 评论

福哥答案2020-09-20:#福大大架构师每日一题#

1.试除法。朴素素数筛,埃氏筛,欧拉筛和区间筛。代码采用朴素素数筛。

2.费尔马素性测试法法。费马小定理:假如p是质数,a是整数,且a、p互质,那么a的(p-1)次方除以p的余数恒等于1,即:a^(p-1)≡1(mod p)。

3.米勒拉宾素性检验法。二次探测定理:如果p是一个素数,0<x<p,则方程x^2≡1(mod p)的解为x=1或x=p-1。

4.综合法。试除法+米勒拉宾素性检验。

5.AKS算法。暂时无代码。

因为用到了大整数,所以用python语言编写。代码如下:

# -*-coding:utf-8-*-

import math
import time
from functools import wraps


def quick_power(a, b, p):
    """
    求快速幂。ret = a^b%p。

    Args:
        a: 底数。大于等于0并且是整数。
        b: 指数。大于等于0并且是整数。
        p: 模数。大于0并且是整数。

    Returns:
        返回结果。

    Raises:
        IOError: 无错误。
    """
    a = a % p
    ans = 1
    while b != 0:
        if b & 1:
            ans = (ans * a) % p
        b >>= 1
        a = (a * a) % p
    return ans


def timefn(fn):
    """计算性能的修饰器"""

    @wraps(fn)
    def measure_time(*args, **kwargs):
        t1 = time.time()
        result = fn(*args, **kwargs)
        t2 = time.time()
        print(f"@timefn: {fn.__name__} took {t2 - t1: .5f} s")
        return result

    return measure_time


@timefn
def is_prime_trial_division(num):
    """
        判断是否是素数。试除法。

        Args:
            num: 大于等于2并且是整数。

        Returns:
            返回结果。true为素数;false是非素数。

        Raises:
            IOError: 无错误。
    """
    if num <= 1:
        return False
    if num == 2 or num == 3 or num == 5 or num == 7:
        return True
    if num % 2 == 0:
        return False
    i = 3
    while num % i != 0:
        if i * i >= num:
            return True
        i = i + 2
    return False


@timefn
def is_prime_fermat(num):
    """
        判断是否是素数。费尔马素性测试法(Fermat primality test) 可能会把合数误判为质数。

        Args:
            num: 大于等于2并且是整数。

        Returns:
            返回结果。true为素数;false是非素数。

        Raises:
            IOError: 无错误。
    """
    if num <= 1:
        return False
    if num == 2 or num == 3 or num == 5 or num == 7:
        return True
    if num % 2 == 0:
        return False
    a = 2  # a是[2,num-1]之间的随机数
    if quick_power(a, num - 1, num) == 1:
        return True
    else:
        return False


# 米勒-拉宾素性检验是一种概率算法,但是,Jim Sinclair发现了一组数:2, 325, 9375, 28178, 450775, 9780504, 1795265022。用它们做 [公式] , [公式] 以内不会出错,我们使用这组数,就不用担心运气太差了。
@timefn
def is_prime_miller_rabin(num):
    """
        判断是否是素数。米勒拉宾素性检验是一种概率算法 可能会把合数误判为质数。

        Args:
            num: 大于等于2并且是整数。

        Returns:
            返回结果。true为素数;false是非素数。

        Raises:
            IOError: 无错误。
    """
    # num=(2^s)*t
    a = 2  # 2, 325, 9375, 28178, 450775, 9780504, 1795265022
    s = 0
    t = num - 1
    num_1 = t
    if not (num % 2):
        return False
    while not (t & 1):
        t >>= 1
        s += 1
    k = quick_power(a, t, num)
    if k == 1:
        return True
    j = 0
    while j < s:
        if k == num_1:
            return True
        j += 1
        k = k * k % num
    return False


@timefn
def is_prime_comprehensive(num):
    """
        判断是否是素数。综合算法:试除法+米勒拉宾素性检验 可能会把合数误判为质数。

        Args:
            num: 大于等于2并且是整数。

        Returns:
            返回结果。true为素数;false是非素数。

        Raises:
            IOError: 无错误。
    """
    if num <= 1:
        return False
    if num & 1 == 0:
        return False

    # 100以内的质数表
    primeList = [3, 5, 7, 11, 13, 17, 19, 23, 29, 31, 37, 41, 43, 47, 53, 59, 61, 67, 71, 73, 79, 83, 89, 97]

    # 质数表是否能整除
    for prime in primeList:
        if num == prime:
            return True
        if num % prime:
            if prime * prime >= num:
                return True
        else:
            return False

    # 米勒拉宾素性检验
    return is_prime_miller_rabin(num)


if __name__ == "__main__":
    print(is_prime_trial_division(12319), "试除法")
    print("----------------------")
    print(is_prime_trial_division(561), "试除法")
    print("----------------------")
    num = 1111111111111111111  # 质数
    num = 561  # 合数
    num = 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFC2F  # 质数
    num = 0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364141  # 质数
    num = 2 ** 10000 + 111  # 合数
    print(is_prime_fermat(num), "费尔马素性测试法")
    print("----------------------")
    print(is_prime_miller_rabin(num), "米勒拉宾素性检验")
    print("----------------------")
    print(is_prime_comprehensive(num), "综合法")
    print("----------------------")
    print("AKS算法,暂时没代码")

执行结果如下:



***

[评论](https://user.qzone.qq.com/3182319461/blog/1600556241)

相关推荐

w7旗舰版系统怎么恢复出厂设置啊

方法一:1、左键单击任务栏开始按钮2、在启动项菜单右侧找到“控制面板”并左键单击3、在打开的界面中找到“区域和语言”选项并左键单击4、在弹出窗口中选择“键盘和语言”,在“选择显示语言”下...

ubuntu下载安装(Ubuntu下载安装包)

要在Ubuntu上从官方网站下载和安装Evolution,您可以按照以下步骤进行操作:1.打开您的网页浏览器,访问Ubuntu的官方网站:https://ubuntu.com。2.点击页面顶部的“...

苹果强制恢复出厂设置(苹果强制恢复出厂设置密码忘记了)
  • 苹果强制恢复出厂设置(苹果强制恢复出厂设置密码忘记了)
  • 苹果强制恢复出厂设置(苹果强制恢复出厂设置密码忘记了)
  • 苹果强制恢复出厂设置(苹果强制恢复出厂设置密码忘记了)
  • 苹果强制恢复出厂设置(苹果强制恢复出厂设置密码忘记了)
win7补丁目录(windows补丁目录)
  • win7补丁目录(windows补丁目录)
  • win7补丁目录(windows补丁目录)
  • win7补丁目录(windows补丁目录)
  • win7补丁目录(windows补丁目录)
联想显示器售后服务电话(lenovo人工客服24小时)

联想显示器保修期限在1~2年之内,一,联想“三包”服务承诺联想按国家有关部门颁布的《微型计算机商品修理更换退货责任规定》(以下称“三包”规定)中的内容和范围,向用户提供“三包”服务。联想承担法定“...

xp系统界面(xp系统界面无图形选项)
  • xp系统界面(xp系统界面无图形选项)
  • xp系统界面(xp系统界面无图形选项)
  • xp系统界面(xp系统界面无图形选项)
  • xp系统界面(xp系统界面无图形选项)
ipad密码忘了怎么办最简单的方法

一般ipad开机密码忘了有以下这种方法可以试一下:操作步骤/方法 1.下载最新版的iTunes。2.通过数据线将ipad与电脑iTunes相连接。3.将ipad按住电源键关机。4.同时按住电...

戴尔官翻机官网(戴尔官翻机购买地址)

肯定可以购买啊,价格还便宜。如果是官翻机应该是可以的,不像市场上的私人翻新机,这个质量应该有保障的可以买的。就是官方翻新机,市场上是有的。具体进入渠道先不管。反正市面上是肯定有的。但是这类手机是享受苹...

手机系统在哪里找(手机系统需要更新吗)

设置方法如下:1、首先输入锁屏密码,进入桌面;2、打开【设置】进入系统设置中心,打开【应用市场】即可查找应用程序;3、进入设置中心的【更多设置】,找到【开发者选项】;4、打开【开启开发者选项...

电脑城装机系统在哪里下载(电脑城都用什么装系统)

其实所有系统基本都一致,装机版已经集成一些必要软件,有个别也许还做了优化,但相对而言,在安全性和兼容性上差一些。不是绝对的,因为装机版也分好多种。纯净版最好,系统干净,使用放心。一般的电脑维修店都能够...

win7右键没有个性化(win7系统右键没有个性化)
win7右键没有个性化(win7系统右键没有个性化)

如果电脑右键没有出现个性化和属性选项,可能是系统出现了错误或者个人设置问题。解决方法如下:首先检查计算机中是否安装有系统文件,如果文件丢失或被删除,则需要重新安装系统文件;其次,可以尝试恢复系统默认配置,去掉不必要的自定义设置;最后,如果上...

2025-12-25 08:03 off999

ie11官方下载(ie11官方免费下载)

1/9点击左下窗户小图标。2/9点击齿轮设置图标。3/9点击【应用】。4/9点击【应用和功能】。5/9点击【程序和功能】6/9点击【启动或关闭windows功能】。7/9ie11没有勾选。8/9点击勾...

东芝笔记本怎么进入bios(东芝笔记本怎么进入语言界面)

以下是一些通用的方法:1.重启或开机时,按下F2键进入BIOS设置界面。在某些笔记本上,您需要按住Fn键才能按F2键。2.重启或开机时,同时按下Ctrl+Alt+Del键,然后立即按F2键...

window10家庭版下载(window10家庭版下载地址)

家庭版用户可通过以下步骤下载AutoCAD2020:访问Autodesk官方网站(autodesk.com)并创建或登录您的Autodesk帐户。导航到AutoCAD2020...

超级吞噬系统txt(超级吞噬系统txt完整版下载)

男主从未推倒柳儿,一直把柳儿当妹妹,出去历练升级从未带着她,后面男主把她安置好后,作者就再没写过她。两人一直分开。吞噬星空的九大超级势力有六大巅峰种族,人族,虫族,机械族,妖族,晶族,狱族,还有另外三...

取消回复欢迎 发表评论: