质数笔记

质数又称素数。一个大于1的自然数,除了1和它自身外,不能被其他自然数整除的数叫做质数;否则称为合数。

([1,N]) 中的素数大概有 (N/InN)

I.素数判断

首先,根据定义,我们可以枚举((1,n)) 里的数,若存在 (i) 使得 (i|n) ,
(n) 是合数,否则 (n) 是质数。
但其实不用枚举到 (n) , 因为 (n) 如果是合数,则其必定有一个小于等于 (sqrt{n}) 的因子。
所以枚举到 ([sqrt{n}]) 即可。

bool ifprime(LL x)
{
	int i;
	for(i=2;i*i<=x;i++) 
		if(x%i==0) 
			return false;
	return true;
}

常数更小的筛法
把每一个数都表示成 (6x+y,(x,yin N)) 的形式。
那么有

  • (6x+0) 可以被 2,3,6 整除
  • (6x+1) 待定
  • (6x+2) 可以被 2 整除
  • (6x+3) 可以被 3 整除
  • (6x+4) 可以被 2 整除
  • (6x+5) 待定

先判断 (n) 能否被 (2,3) 整除(假定$n in [4,+infty] $)

  • 若能则 (n) 不是质数
  • 否则枚举 (x in {x|x=6k+1,kin Z} cup {x|x=6k+5,kin Z}) ,判断 (n) 能否整除 (x)
  • (n in [1,3]) 时,特判掉即可。
bool isprime(LL x)
{
    if(x==2||x==3) return 1;
    if(x%6!=1&&x%6!=5) return 0;
    for(int i=5;i*i<=x;i+=6)
        if(x%i==0||x%(i+2)==0)
        	return 0;
    return 1;
}

上述算法的实际运行次数大约为 $ sqrt{n} div 6$

II.素数筛法

1.朴素筛法

时间复杂度(O(n^{3/2}))

很容易想到从1枚举到n,挨个判断是不是素数;

Code:
void Prime()
{
	int i,j;
	for(i=2;i<=n;i++) {
		if(isprime(i)) 
			pm[++tot]=i;
	}
	return;
}

isprime 函数可参考素数判断的内容。

2.简单筛法

令A为素数,则A*N(N>1;N为自然数)都不是素数,标记为合数。

时间复杂度(O(nlogn))

Code:
const LL N=1e7+5;
int pm[N];
bool tag[N];
int tot;
void Prime()
{
	int i,j;
	for(i=2;i<=n;i++) {
		if(!tag[i]) 
			pm[++tot]=i;
		for(j=2*i;j<=n;j+=i) 
			tag[j]=true;			
	}
	return;
}

优点:稍微改造一下可以标记出 ([1,n]) 中所有数的约数。

缺点:时间复杂度。

推论:([1,n]) 中所有数的约数数量之和约为 (nlogn) .

3.Eratosthenes筛法

(埃拉托色尼斯筛法)

埃氏筛法其实是对 2 的优化。

  1. (x) 已经被标记为合数,则不用 (x) 去标记其他数。

    • 据理:(x) 是一个合数,则 (x) 起码有两个质因子 (y),(z)
    • (y,z<x) , 观察上述算法流程,可发现所有能被 (x) 筛掉的数已经被 (y,z)(x) 的其他质因子筛掉一遍了。
    • 值得一提的是,单用(2和优化1)时,可以标记出 ([1,n]) 中所有数的质因数。
  2. 二是 (i) 若为素数,则从 (i*i) 开始标记合数。

    • 据理:对于 任意 (x in (i,i*i)) ,若 (i|x) ,则 (x) 必有另一个约数 (<i) .
    • 证明: (x/i)(x) 的一个约数,且其比 i 小。
    • 所以: ((i,i*i)) 里的数已经被比i小的数标记过了,(i)(i*i) 开始标记。

Code:

void prime()
{
    LL i,j;
    for(i=2;i<=n;i++)
    {
    	if(tag[i])  continue;   //优化1
        pm[++tot]=i;
        for(j=i*i;j<=n;j=j+i)   //优化2
        	tag[j]=true;
    }
}

上述算法的时间复杂度为:(O(nloglogn)).

证明在此:https://www.zhihu.com/question/35112789。

图片来自:https://zhuanlan.zhihu.com/p/48560969

4.线性筛法---Euler筛

观察Eratosthenes筛法的动图,我们发现有些数比如110,还是被筛了多次。

我们不妨规定一个顺序,让每个合数只被他的最小质因子筛掉。

一个形象的说法就是,对于每一个数,配合比他的最小的一个质因子还小的质数 (x),筛掉合数。

((x) 筛掉的合数 的最小质因子 就是 (x))

这样每个数都只会被其最小的质因子筛掉一次。

时间复杂度(O(n))

Code:
void Prime()
{
	int i,j;
	for(i=2;i<=n;i++) {
		if(!tag[i]) 
			pm[++tot]=i;
		for(j=1;j<=tot;j++) { //配合其他质数
			if(i*pm[j]>n) break; //越界
			tag[i*pm[j]]=true; // 标记合数
			if(i%pm[j]==0) // 若pm[j]|i,那么 pm[j+1]*i 的最小质因子就是 pm[j] 而不是 pm[j+1] 了
				break;  // 违反的上述规定,于是就break掉
		}               // 关于 pm[j+1]*i ,不用担心,他会被 pm[j] 配合其他某个数筛掉 
	}                   // 这样就做到了不重不漏
	return;
}

END

1000000以内的质数请见:https://cjlworld.blog.luogu.org/zhi-shuo-biao

(10^{10}) 以内的质数请见:

原文地址:https://www.cnblogs.com/cjl-world/p/13381306.html