线性筛素数

#include<cstdio>
#define MAXN 10000
using namespace std;
bool prime[MAXN];
void Prime() {//时间复杂度O(n*lglgn)
	for (int i = 2; i < MAXN; i++) prime[i] = 1;//先将所有的数设为质数
	for (int i = 2; i < MAXN; i++) {
		if (prime[i])		//对于质数,用它去筛掉它的倍数,它的倍数均为合数
			for (int j = i * 2; j < MAXN; j += i)
				prime[j] = 0;
	}
}//此方法一个数会被多次筛掉,比如6会被2和3各筛一遍,重复计算

bool vis[MAXN];
int pri[MAXN];//vis表示合数集合,pri表示素数集合
void linearPrime() {//时间复杂度接近O(n)
	int cnt = 0;
	for (int i = 2; i < MAXN; i++) {
		//因为i是增加的,所以在下一行代码,若i是素数,则内层for循环不会中断
		//所有素数的i倍都会被筛掉,因为i是新加入的素数,所以是目前最大的素数,即那些被筛掉的数的最大质因数就是i
		if (!vis[i]) pri[cnt++] = i;	//若此数不是合数,则加入素数集合
		for (int j = 0; j < cnt&&i*pri[j] < MAXN; j++) {//遍历素数集合,把它们的i倍记为合数
			vis[i*pri[j]] = 1;
			if (i%pri[j] == 0) break;	//为使速度最快,应力求对每个数只判断一次,所以应该由该数最大的质因数筛掉,因此在遇到该数的较小质因数时应跳过
			//若中断,则i是合数,所有数应该被它的最大质因数筛掉,而不是被较小的质因数或者合数筛掉,才能保证每个数只做一次判断
		}
		
	}
}
int main() {
	Prime();
	for (int i = 2; i < MAXN; i++)
		if (prime[i])
			printf("%d ", i);

	linearPrime();
	int i=0;
	while (pri[i])
		printf("%d ", pri[i++]);
	return 0; 
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值