关于质数(素数)

栏目: Python · 发布时间: 7个月前

来源: juejin.im

本文转载自:https://juejin.im/post/5c19d072e51d45490253100a,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有。

质数(Prime number)又称素数, 质数是大于 而且只能被 和自身整除的自然数。大于 的自然数如果不是素数,就称为合数(Composite number)。

算术基本定理

算术基本定理最早由欧几里得证明, 是表示任何合数都可以不断分解成素数的组合,如 可以分解为 两个素数,欧几里得发现把这些素数因子的次方相乘可以得到原来的数字 ,而且这种分解为素数乘积的方式是唯一的。素数因子分解就像是一个锁,而且只有一把开锁的钥匙,这也是现代密码学的基础。

素数定理

素数定理描述素数在自然数中分布的渐进情况,就是小于 中素数的个数随着 的增大素数的密度就越来越小。当 越来越大时它的图像就越来越接近 。所以一个数字内素数的数量 约等于

试除法

那么怎么判断一个数是不是素数?这也是很多面试题里面问到的。一种简单的方法是试除法。

比如判断 是不是素数,可以让 除以 到 之间的整数,如果可以除尽则表示是合数否则是质数。

def isPrime(n):
    if n < 2: return False
    for i in range(2, n):
        if n % i == 0:
            return False
    return True
复制代码

但是可以发现大于 偶数都不是质数,因为它们可以被 整除,所以这可以减少迭代次数。

def isPrime(n):
    if n == 2: return True
    if n < 2 or n % 2 == 0: return False
    for i in range(3,n,2):
        if n % i == 0:
            return False
    return True
复制代码

还有没有更快的方法?有,就是迭代到 ,因为一个数字分解为两个因子,其中必然有一个小于或等于 ,不然两个都大于 它们相乘就大于 了。

def isPrime(n):
    if n == 2: return True
    if n < 2 or n % 2 == 0: return False
    for i in range(3, int(n ** 0.5) + 1, 2):
        if n % i == 0:
            return False
    return True
复制代码

试除法还能不能更快?根据算术基本定理任何合数最终都可以分解为素数的组合,所以只用除小于 的素数就行了。

筛选法

筛选法可以给出小于 的素数序列,比如要生成 内的素数序列,首先可以生成2到100间数字表,然后将列表第一个没被标记的数字标记为素数然后将数字表中它的倍数标记为合数,然后不断重复这个步骤。

关于质数(素数)
def genPrimes():
    primes = [2]
    i = 1
    yield 2
    while True:
        i += 2
        for p in primes:
            if i % p == 0:
                break
        else:
            primes.append(i)
            yield i
复制代码

费马小定理

虽然一般判断是不是素数用 试除法 就行了,但是当要判断一个大数是不是素数时,也还是太慢了。

费马小定理 是欧拉定理的一个特殊情况,它是说一个正整数 的 质数 次方减 可以被 次方整除。用公式表示可以为 。

但是也不能完全正确,比如 但是 ,所以可以随机生成多个 来测试,这样就可以降低将出错概率。

import random
def gcd(a, b):
    while b != 0:
        a, b = b, a % b
    return a
def randA(p):
    return random.randint(1, p - 1)
def fastMod(f, p, m):
    ans = 1
    whlie p > 0:
        if p % 2 == 1:
            ans = (ans * f) % m
            p -= 1
        p //= 2
        f = (f ** 2) % m
    return ans
def isPrime(p):
    trials = 30
    for i in range(trials):
        a = randA(p)
        if gcd(a, p) != 1: return False
        if fastMod(a, p - 1, p) != 1: return False
    return True
复制代码

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持 码农网

为你推荐:

相关软件推荐:

查看所有标签

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

总开关

总开关

[美] 吴修铭 / 顾佳 / 中信出版社 / 2011-8 / 49.00元

当收音机经历从真空管收音机到半导体收音机,再到电晶体收音机的发展升级时,人们觉得自己的资讯来源美满得无可复加了。当约翰.洛吉.贝尔德发明了电视以后,在很长一段时间内,人们都认为电视就是他们所拥有的﹑也是所愿意拥有的最好的资讯媒介。 时至今日,互联网的震撼不亚于以往任何媒介,它给我们带来了最大的信息量,最便捷的自我表达,最迅速的沟通。互联网似乎比以往任何媒介都具有优越性。在互联网成为这个时代主......一起来看看 《总开关》 这本书的介绍吧!

HTML 编码/解码
HTML 编码/解码

HTML 编码/解码

MD5 加密
MD5 加密

MD5 加密工具

HEX HSV 转换工具
HEX HSV 转换工具

HEX HSV 互换工具