`

SPOJ Prime Generator

    博客分类:
  • SPOJ
阅读更多
SPOJ   Prime Generator
   题目意思:给定两个正整数m,n,(1<=m<=n<=10^9,n-m<=100000),求m到n之间的所有素数。
  方法1:对区间[m,n]内的每一个数进行miller_rabin素数测试(时间复杂度较高,约为1.37s~4.6s)
  代码实现所需的算法有:
① 产生[0,1]及[0,m-1]之间的随机数函数Random(),Random(m)
② 手写乘法取模函数mul_mod()
③ 二分求幂a^b%n函数pow_mod(a,b,n)
④ Miller_rabin素数测试算法
#include <cstdio>
#include <time.h>
#include <cstdlib>
#include <iostream>
using namespace std;
typedef long long lld;
//产生[0,1]之间的随机数
double Random()
{
    return (double)rand()*1.0/RAND_MAX;
}
//产生[0,m-1]之间的随机数
lld Random(lld m)
{
    return (lld)(Random()*(m-1)+0.5);
}
//手写乘法取模,防止64位溢出
//注意此函数最好是在用__int64还会溢出的情况下使用
//否则在其它情况下会增加程序的时间复杂度
//如在SPOJ Prime Generator题目中使用就TLE,去掉就AC啦
//同时也仍需要注意在实现此函数应少用取模运算%,因为这也会增加
//程序的时间复杂度,在某些题目中也会TLE,如FZU A^B%C
lld mul_mod(lld a,lld b,lld n)
{
    lld res=0,temp=a%n;
    while(b)
    {
        if(b&1)
        {
            res+=temp;
            if(res>=n)  res-=n;
        }
        temp<<=1;
        if(temp>=n) temp-=n;
        b>>=1;
    }
    return res;
}
//求a^n%p
//注意在SPOJ Prime Generator中需注释mul_mod
//如果a*b%n用__int64仍会溢出则必须用mul_mod
lld pow_mod(lld a,lld n,lld p)
{//二分求幂
    lld res=1,half=a%p;
    while(n)
    {
        if(n&1) res=res*half%p;//mul_mod(res,half,p);
        half=half*half%p;//mul_mod(half,half,p);
        n>>=1;
    }
    return res;
}
//判断n是否是合数,如果是则返回true,否则false
/*bool witness(lld a,lld n)
{//a^(n-1)=1 (mod n)
    lld u=n-1;
    lld t=0;
    while((u&1)==0)
    {//计算n-1=u*(2^t),其中t>1,u为奇数
        u>>=1;
        t++;
    }
    lld x=pow_mod(a,u,n);
    for(int i=1;i<=t;i++)
    {
        lld k=mul_mod(x,x,n);
        if(k==1&&x!=1&&x!=n-1)  return true;
        x=k;
    }
    if(x!=1)    return true;
    return false;
}*/

bool witness(lld a, lld n)
{
    lld m = n - 1;
    lld q = 0;
    while((m&1) == 0)
    {
        q ++;
        m >>= 1;
    }
    lld x = pow_mod(a, m, n);
    if (x == 1 || x == n-1) return false;//n可能为素数
    while(q --)
    {
        x = x * x % n;
        if (x == n-1) return false;
    }
    return true;//n一定是合数
}


bool miller_rabin(lld n,lld s)
{//此函数的时间复杂度可以通过调整s的大小来调节,一般s>=3
    if(n==2||n==3)  return true;
    if(n<=1||n%2==0||n%3==0)    return false;
    for(int i=1;i<=s;i++)
    {
        lld a=Random(n-1)%(n-1)+1;
        if(witness(a,n))    return false;
    }
    return true;
}
int main()
{
    int t;
    lld m,n;
    srand(time(NULL));
    scanf("%d",&t);
    while(t--)
    {
        scanf("%lld%lld",&m,&n);
        for(lld i=m;i<=n;i++)
        {
            if(miller_rabin(i,20))  printf("%lld\n",i);
        }
        printf("\n");
    }
    return 0;
}

方法二:这题使用的是标准筛法的一个修改版本。如果用普通的筛法,对[2, n]内的每个数进行排查,明显不是高效的,会使用很多的时间和空间。然而,我们可以发现,[0, n]内没有一个合数的因子大于floor(sqrt(n))。所以,我们只需要把[2, sqrt(n)],即[2, 31622]以内的素数筛出来。然后,对于每个询问[a, b],使用预先筛好的素数进行第二次筛选,最后得到[a, b]内的素数,输出。
涉及的算法:
① 素数筛选法及二次筛选给定区间[m,n]内的素数
代码1:
#include <iostream>
#include <cstdio>
#include <memory.h>
#define MAXN 32000
using namespace std;
bool a[MAXN];
int prime[MAXN];//储存2~32000内所有的素数
int num;//记录素数的个数
int m,n;
int p[100010];
void getPrime1()
{//素数筛选法 
    memset(a,0,sizeof(a));// 初始进假设所有数均为素数
    a[0]=a[1]=1;
    for(int i=2;i*i<=MAXN;i++)
    {
        if(!a[i])
        for(int j=i;j*i<=MAXN;j++)
        a[j*i]=1;
    }
    num=0;
    for(int i=0;i<MAXN;i++)
    {
        if(!a[i])   {prime[num++]=i;
        //cout<<i<<" ";
        }
    }
    //cout<<endl;
}
void getPrime2()
{//二次筛选把[m,n]内的合数删除 
    memset(p,0,sizeof(p));
    for(int i=0;i<num&&prime[i]<=n;i++)
    {
        //cout<<prime[i]<<" ";
        int k=m/prime[i];
        for(int j=k;j*prime[i]<=n;j++)
        {
            if(j!=1&&j*prime[i]>=m) p[j*prime[i]-m]=1;
        }
    }
    for(int i=0;i<=n-m;i++)
    {
        if(!p[i]&&i+m!=1)   printf("%d\n",i+m);
    }
}
int main()
{
    int t;
    num=0;
    getPrime1();
    scanf("%d",&t);
    bool flag=false;
    while(t--)
    {
        if(flag)    printf("\n");
        flag=true;
        scanf("%d%d",&m,&n);
        getPrime2();
    }
    return 0;
}
分享到:
评论

相关推荐

    SPOJ.rar_SPOJ

    《SPOJ在线判题平台:越南版解题攻略》 SPOJ,全称Sphere Online Judge,是一个全球知名的在线编程竞赛平台,旨在提供一个环境让程序员可以测试、提交并评估他们的算法解决方案。这个名为"SPOJ.rar_SPOJ"的压缩包...

    SPOJ极少量的源程序。

    3. **PRIME1**:题目很可能与素数有关。需要了解素数的定义,可能需要实现快速检测素数的算法,如埃拉托斯特尼筛法或米勒-拉宾素性测试。 4. **PALIN**:这可能是指回文字符串的检测。需要理解字符串操作,如字符串...

    spoj做题表格

    标题:“spoj做题表格”与描述“杨弋SPOJ做题表格”共同指向了在SPOJ(Sphere Online Judge)平台上的编程题目列表,这通常被用于记录和跟踪解决算法问题的进度。SPOJ是一个知名的在线裁判系统,为全球范围内的...

    SPOJ-2.zip_SPOJ_SPOJ-TRIKA.cpp_online judge_sPOJ-Solution_spoj2

    《SPOJ在线判题平台的解题策略与实例解析》 SPOJ(Sphere Online Judge)是一个全球知名的在线编程竞赛平台,它为程序员提供了一个检验自己算法能力的场所。这个名为"SPOJ-2.zip"的压缩包包含了在SPOJ上解决不同...

    spoj 第42题 reverse

    spoj reverse 的solution

    SPOJ题库-离线题库-索引+内容+PDF

    SPOJ题库( http://www.spoj.pl)的离线题库。 包括索引+内容。PDF格式。 主要是Classical的problemset。

    SPOJ.zip_SPOJ_sgu_them

    《SPOJ问题解决策略:动态规划的应用》 在编程竞赛和算法研究中,SPOJ(Sphere Online Judge)是一个著名的在线平台,它提供了一系列挑战性的编程问题供参赛者解决。"SPOJ.zip_SPOJ_sgu_them"这个压缩包文件就包含...

    acm pku spoj sgu 经典 图论题解题报告

    hdu 1914稳定婚姻 sgu176 有源汇的上下界 求最小满足的流 poj 2230 递归求欧拉回路 poj 2985 bst模板 poj2723 2-sat验证,二分答案 poj2455 dinic (ek会超时) hdu1689 求最小奇数环 poj2391 isap最快,dinic不减枝...

    CPP.zip_SPOJ-HOTELS.cpp_circular game spoj_cpp_segment tree_usac

    标题 "CPP.zip_SPOJ-HOTELS.cpp_circular game spoj_cpp_segment tree_usac" 提到了几个关键元素,包括一个编程挑战(SPOJ-HOTELS)、特定的编程语言(C++)、一种算法(circular game)以及数据结构(segment tree...

    SPOJ:我对 SPOJ 问题的解决方案 (www.spoj.com)

    SPOJ,全称为Sphere Online Judge,是一个在线的编程竞赛平台,它提供了各种算法和数据结构问题供用户解决,以提升编程技能和算法理解。在这个平台上,你可以使用多种编程语言,包括C++,来编写代码并提交解决方案。...

    SPOJ-Solutions:SPOJ算法问题的解决方案

    SPOJ(Sphere Online Judge)是一个在线编程竞赛平台,它提供了大量的算法问题供程序员解决,以提高他们的编程技能和算法理解。"SPOJ Solutions"是这个领域的一个资源集合,包含了解决SPOJ上各种问题的代码示例,...

    cp:一些SPOJ问题的解决方案

    标题 "cp:一些SPOJ问题的解决方案" 暗示了这是一个关于计算机编程,特别是针对SPOJ(Sphere Online Judge)平台上的算法问题解决的资源。SPOJ是一个在线的编程竞赛平台,用户可以在这里提交代码来解决各种算法问题,...

    SPOJ:Python中SPOJ问题的解决方案

    在编程竞赛领域,SPOJ(Sphere Online Judge)是一个广受欢迎的在线判题系统,它提供了大量的编程题目供参赛者解决,以提升算法和编程能力。对于Python爱好者来说,掌握如何在SPOJ上高效地解决问题是至关重要的。...

    SPOJ-BACKUP-TOOL:在 SPOJ 上下载已提交代码的脚本

    SPOJ-备份工具 介绍 在 Sphere Online Judge ( ) 中,您可以尝试所给的具有挑战性的问题。 它还使您能够查看和下载您自己的解决方案。 工具 SPOJ_BACKUP 备份所有已接受的提交并将它们保存在脚本所在的计算机位置。...

    Problem Set 1.zip_FFT simulink_SPOJ

    标题中的"Problem Set 1.zip_FFT simulink_SPOJ"揭示了这是一个关于傅立叶变换(FFT)的项目,具体来说是8点快速傅立叶变换(8-point Fast Fourier Transform),并且它与MATLAB仿真工具Simulink以及在线编程挑战...

    spoj-solutions:简单的SPOJ问题的解决方案。 看

    标题中的"spoj-solutions"指的是Sphere Online Judge (SPOJ) 的解决方案集合。SPOJ是一个在线的编程竞赛平台,它提供了各种算法和数据结构问题供用户解决,以提高编程技能和解决问题的能力。"简单的SPOJ问题的解决...

    spoj4491 莫比乌斯反演

    ### spoj4491 莫比乌斯反演 #### 题目解析 题目要求计算在给定的两个正整数\( n \)和\( m \)范围内,有多少对\((a, b)\)使得它们的最大公约数\(\text{gcd}(a, b) = d\),其中\(d\)为素数,并且\(1 \leq a \leq n\)...

    Ultimate SPOJ Rank Tracker-crx插件

    分机检查spoj用户的排名。 此扩展名有2个选项: - 在选择的比赛中显示用户排名 - 在选择比赛中最多可比较5个用户第一个在后台运行,并在每隔几分钟内更新,您可以设置自己(并默认为15分钟)。当您单击分机的徽标时...

    MSTS.rar_ MSTS-CN_MSTS-CN_SPOJ_msts

    本主题将深入探讨Kruskal算法及其在SPoj(Sphere Online Judge)在线编程平台上的应用。 首先,让我们了解什么是MSTS(Minimum Spanning Tree)。在无向加权图中,MSTS是一棵包括所有顶点的树,其边的权重之和尽...

    sublex.tar.gz_SUBLEX_spoj SUBLEX

    《SUBLEX与后缀自动机在SPOJ平台的应用》 SUBLEX是SPOJ平台上一个关于后缀自动机(Suffix Automaton)的挑战,这个题目旨在考察程序员对后缀自动机的理解以及实现能力。后缀自动机是一种非常重要的字符串处理工具,...

Global site tag (gtag.js) - Google Analytics