筛表合集(素数筛 欧拉函数筛 莫比乌斯函数筛)

【目录】

一、素数筛

1.素数判断

2.素数普通筛

3.素数线性筛

4.素数区间筛

二、欧拉函数筛

三、莫比乌斯函数筛

 

 

【素数筛】

1.直接判定质数

bool judgePrime( int num )
{
    if(num < 2 )    return 0;
    int tmp =sqrt( num );
    for(int i= 2; i <=tmp; i++)
        if(num % i == 0)
            return 0 ;
    return 1 ;
}

 

2.素数普通筛

const int maxn = 1e7;
bool isPrime[maxn+10];
int prime[maxn+10];
int count = 0;


//普通筛法 
void getPrime(){
    memset(isPrime, 1, sizeof isPrime);
    isPrime[1] = isPrime[0] = 0;
    //筛出bool数组 
    for(int i=2; i*i<=maxn; i++){ //如果想在这里直接把素数打入prime数组里那么这里i<maxn 
        if(isPrime[i]){
            for(int j=2; j<=maxn/i; j++){
                isPrime[i*j] = 0;
            }
        }
    }
    //根据bool数组把素数打入素数表 ,如果只要判断素数则以下代码可注释 
//    for(int i=0; i<maxn; i++){
//        if(isPrime[i])
//            prime[count++] = i;    
//    }    
}

 

3.素数线性筛

保证每个素数只会被筛一次

//线性筛法
bool check[maxn + 10];
int prime2[maxn + 10];
int count2 = 0; 
void getPrime2(){
    memset(check, 1, sizeof check);
    check[0] = check[1] = 0;
    for(int i=2; i<=maxn; i++){
        if(check[i]){
            prime2[count2++] = i;    
        }
        for(int j=0; j<count2; j++){
            if(i*prime2[j] > maxn) break;
            check[i*prime2[j]] = 0;
            if(i%prime2[j] == 0) break;
        } 
    }    
} 

 

4.素数区间筛

给定整数a和b,请问区间[a,b]内有多少个素数? 

a<b<=10^12

b-a<=10^5

因为b以内合数的最小质因数一定不超过sqrt(b),如果有sqrt(b)以内的素数表的话,就可以把筛选法用在[a,b)上了.

先分别做好[2,sqrt(b)]的表和[a,b]的表,然后从[2,sqrt(b)]的表中筛得素数的同时,也将其倍数从[a,b)的表中划去,最后剩下的就是区间[a,b]内的素数了。

#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long ll;

const int maxn = 2e5+10;

bool isPrimeSmall[maxn];
bool isPrimeBig[maxn];

ll  getInterval(ll a, ll b){
    memset(isPrimeSmall , 1 , sizeof isPrimeSmall);
    memset(isPrimeBig , 1 , sizeof isPrimeBig);
    isPrimeSmall[0] = isPrimeSmall[1] = 0;
    for(ll i = 2; i*i<b; i++){
        if(isPrimeSmall[i]){
            for(ll j=2*i; j*j<b; j+=i){
                isPrimeSmall[j] = 0;
            }
            for(ll k = max(2LL, (a+i-1) / i) * i; k<=b; k+=i){
                isPrimeBig[k - a] = 0;
            }
        }
    }
    ll ans = 0;
    for(ll i=0; i<= b-a; i++){
        if(isPrimeBig[i])    ans ++;
    }
    
    if(a == 1)    ans--;//防止1被当成质数 
    return ans;
}



int main()
{
    ll a,b;
    int t;
    scanf("%d",&t);
    int cas = 1;
    while(t--)
    {
        scanf("%lld %lld",&a,&b);
        printf("Case %d: %lld\n",cas++, getInterval(a,b));
    }
    return 0;
}

 

小结:当需要频繁地判定素数时需要筛表,否则可以直接进行素数判断。筛表时,如果需要频繁地用素数而不是仅仅需要判定素数,用线性筛,否则用普通筛筛出bool数组即可。

筛表的范围,1e6基本无压力,1e7稍微有点大但能撑得住,1e8筛表速度很慢,1e9无法筛表,时间空间都跟不上。

 注意:如果出现runtime error,可能是

int prime[maxn+10] 数组过大,可以先手动看一下count的值定这个数组的空间。

二、欧拉函数

欧拉函数的定义:

    在数论中,对于正整数N,少于或等于N ([1,N]),且与N互质的正整数(包括1)的个数,记作φ(n)。

     φ函数的值:

     φ(x)=x(1-1/p(1))(1-1/p(2))(1-1/p(3))(1-1/p(4))…..(1-1/p(n)) 其中p(1),p(2)…p(n)为x的所有质因数;

 

1.直接判断

int Euler(int n){
    int ans = n;
    //这里的n在不断收缩  循环次数上界将越来越小 
    for(int i=2; i<=n; i++){
        if(n % i == 0){
            ans = ans - ans/i; 
        }
        //把该素因子除尽 
        while(n % i == 0)    n = n/i;
    }
    return ans;
}

 

 

2.欧拉函数筛表

筛表不太好理解,可以类比质数筛

(1)先锁定欧拉函数计算方法φ(x)=x(1-1/p(1))(1-1/p(2))(1-1/p(3))(1-1/p(4))…..(1-1/p(n)) 其中p(1),p(2)…p(n)为x的所有质因数;

(2)不妨初始化φ(x)=x

(3)每次遇到素因子的时候就乘上一个 p(i)-1 / p(i),p(i)是x的质因子

#define Max 1000  
int euler[Max];  
void Init(){   
     euler[1]=1;  
for(int i=2;i<Max;i++) euler[i]=i; //初始化φ(x)=x for(int i=2;i<Max;i++) if(euler[i]==i) //这个欧拉值没有被更新过则需要更新,保证了进行更新时,i一定是素数 for(int j=i;j<Max;j+=i) euler[j]=euler[j]/i*(i-1);//先进行除法是为了防止中间数据的溢出 }

 

小结:

 

 

三、莫比乌斯函数筛

 【莫比乌斯函数】µ(d)参考博客:https://blog.csdn.net/lixuepeng_001/article/details/50577932

【重要性质】

 

 【反演公式】

【莫比乌斯函数线性筛模板】

const int maxn=1000010;

bool vis[maxn];
int prim[maxn];
int mu[maxn];
int cnt;

void get_mu(){
    mu[1]=1;
    for(int i=2;i<maxn;i++){
        if(!vis[i]){
            prim[++cnt]=i;
            mu[i]=-1;
        }
        for(int j=1;j<=cnt && prim[j]*i<maxn;j++){
            vis[prim[j]*i]=1;
            if(i%prim[j]==0) break;
            else mu[i*prim[j]]=-mu[i];
        }
    }
}

 

【例题】HDU 1695

 

【AC代码】

#include<iostream>
#include<cmath>
#include<algorithm>
using namespace std;
typedef long long ll;
const int maxn=1000010;

bool vis[maxn];
int prim[maxn];
int mu[maxn];
int cnt;

void get_mu(){
    mu[1]=1;
    for(int i=2;i<maxn;i++){
        if(!vis[i]){
            prim[++cnt]=i;
            mu[i]=-1;
        }
        for(int j=1;j<=cnt && prim[j]*i<maxn;j++){
            vis[prim[j]*i]=1;
            if(i%prim[j]==0) break;
            else mu[i*prim[j]]=-mu[i];
        }
    }
}

ll a,b,c,d,k;
int main(){
    int t;
    cin>>t;
    get_mu();
    int cas = 1;
    while(t--){
        cin>>a>>b>>c>>d>>k;
        if(k > d || k > b || k == 0){
            cout<<"Case "<<cas++<<": "<<0<<endl;
            continue;
        }
        b = b/k;
        d = d/k;
        ll ans1 = 0;
        ll ans2 = 0;
        for(int i=1; i<=min(b,d); i++){
            ans1 += mu[i]*(b/i)*(d/i);
        }
        for(int i=1; i<=min(b,d); i++){
            ans2 += mu[i]*(min(b,d)/i)*(min(b,d)/i);
        }
        cout<<"Case "<<cas++<<": "<<ans1 - (ans2>>1) <<endl;
    }
}
 
View Code

 

posted @ 2018-10-09 21:00  西风show码  阅读(481)  评论(0编辑  收藏  举报