zoukankan      html  css  js  c++  java
  • 利用辛达拉姆筛进行素数判定

    1 常规判定方法

           素数判定问题就是对给定的正整数n判定是否为素数。所谓素数。是指恰好有2个约数的整数。由于n的约数都不超过n,所以仅仅须要检查2~n-1的全部整数是否整除n就能判定是不是素数。只是,我们还能进一步优化。假设d是n的约数,那么n/d也是n的约数。由n=d*n/d可知min(d,n/d),所以仅仅须要检查2~的全部整数就足够了。此时,素数判定的复杂度为O()。代码实现例如以下:

    int classic(long long a)
    {
        long long half=(long long)sqrt(a);
    
        for(int i=2;i<=half;i++)
        {
            if(a%i==0)  return 0;
        }
    
        return 1;
    }

    常规的素数判定思想和实现都很easy,可是效率较低。如今存在不少更高效的素数判定方法,比方费马測试、算法、数域筛法等。可是这些算法要么不精确要么须要的数学知识过于复杂,在此不做过多介绍。

    以下我们介绍一种仅仅须要初等数学就可以明确的素数判定方法,该方法在不论什么情况下都会获得比常规方法好一倍的性能。

    2 埃氏筛法

           在介绍新方法之前。我们先回想一下埃氏筛法,这是一个与辗转相除法一样古老的算法。

           大约在公元前3世纪,古希腊数学家埃拉托色尼提出了一种编造素数表的方法(如表一)。

    这样的方法类似于筛东西,把不要的筛掉。把须要的留下。详细做法是:将从2到N的自然数,按顺序排列成2,3,4,5,…,N,然后留下第一个2。划去全部2的倍数;2之后没划去的第一个数是3。留下3,划去全部3的倍数;在3后面没划掉的第一个数是5,留下5,划去全部5的倍数。如此继续。直至上述一列数中再也没有可划的数为止,留下来的便是N以内的一切素数。


    表一 埃氏筛法

           假设仅仅对一个整数进行素性判定,常规判定方法或者后面介绍的方法就足够了。

    但假设要对很多整数进行素性判定,就须要採用埃氏筛法。

    下面枚举算法摘自《挑战程序设计竞赛》一书:

    int prime[MAX_N];
    bool is_prime[MAX_N];
    
    //返回n以内的素数个数
    int sieve(int n) {
    	int p=0;
    	for(int i=0;i<=n;i++) is_prime[i]=true;
    	is_prime[0]=is_prime[1]=false;
    	for(int i=2;i<=n;i++) {
    		if(is_prime[i]) {
    			prime[p++]=i;
    			for(int j=2*i;j<=n;j+=i) is_prime[j]=false;
    		}
    	}
    	return p;
    }
    

    3 辛达拉姆筛

           公元1934年。一名年轻的东印度数学生辛达拉姆。提出了一种与埃拉托色尼迥然不同的筛法。它首先列出了一张表,如表二。表的第一行和第一列都是首项为4。公差为3的等差数列。从第二行開始。以后各行也是等差数列,公差分别为5,7,9,11,13……。能够看出,该表事实上是一个对称矩阵。

           辛达拉姆指出:假设N出如今表中,则2N+1是合数。若N不在表中,则2N+1是素数

    证明相当精彩!

    表二 辛达拉姆筛构造的表

           先证明前半部分。

    首先。他写出来第m行的第一个数:

    4+(m-1)×3=3m+1

    注意到该行是公差为2m+1的等差数列,所以此行第n列的数是:

    3m+1+(n-1)*(2m+1)=2mn+m+n

    即第m行n列的数是2mn+m+n。于是2N+1=4mn+2m+2n+1=(2m+1)(2n+1)是合数。

           再证后半部分。要想正面证明2N+1是素数是相当困难的。假设换成等价的逆否命题,即证“若2N+1不是素数,则N必在表中”似乎要easy得多。

    其实。若

           2N+1=x*y(x ,y)为整数

           则因2N+1为奇数,x、y也必为奇数。最好还是设:

           x=2p+1;y=2q+1

           从而2N+1=(2p+1)(2q+1)=4pq+2p+2q+1。由此能够得到

           N=2pq+p+q。

    也即N是表中第p行第q列的数。

           综上所述。我们证明了辛达拉姆筛法的正确性。比如18不在表中,则2*18+1=37为素数。

    相反,71在表中,则2*71+1=143是合数。后半部分的证明是反证法的绝佳实例。

    4 基于辛达拉姆筛的素数推断   

           有了辛达拉姆筛法,我们该怎样判定一个正整数n是否为素数呢?我们仅仅须要推断(n-1)/2是否在表中就可以。

    因为须要判定的数肯定是奇数(偶数肯定都是合数)。所以(n-1)/2肯定能够整除。推断是否在表中,也即推断是否存在p和q使得2pq+p+q=(n-1)/2。

    依照这个思路实现的代码例如以下:

    int xindalamu(long long a)
    {
    	if(a%2==0) return 0;
    
        long long aa=(a-1)>>1;
        long long half=(long long)sqrt(aa/2);
    
        for(int i=1;i<=half;i++)
        {
            if((aa-i)%(2*i+1)==0) return 0;
        }
    
        return 1;
    }
    

    以下详解一下上面的代码。首先推断用户输入是否是偶数。假设是直接返回false。然后将用户输入减一除以二赋值给aa,后面就须要推断aa是否在表中。推断的方法和常规推断的方法类似。也是须要验证某一行是否出现整除的情况,只是这里公式稍有不同。

    从上述代码能够看到。在对aa开方的时候,我们将其除以二。

    这是为什么呢?推断aa是否出如今表中,也即推断是否存在p和q使得2pq+p+q=aa。

    这和常规方法中的推断aa=p*q稍有不同。忽略掉两个小项,我们须要推断2pq是否等于aa。也即pq是否等于aa/2,这就是除以2的原因。for循环从第一行開始检查。推断该行是否存在列满足2pq+p+q=aa。当行号固定之后。推断的方法是看aa-p是否整除2p+1(稍作变化就可以看出)。

           我们分析一下基于辛达拉姆筛的素数判定复杂度。由for循环能够看出。复杂度也是O()。只是常系数比常规方法要小。这是由于,在求aa时有个除以二的操作,在开方时又有一个除以二的操作。所以相当于对a/4进行开方,因而最多仅仅须要对/2的行进行推断。

    而常规方法最多则须要验证个不同的因子。所以能够看出基于辛达拉姆筛的素数判定方法比常规方法快一倍。

           可是,另一个疑问须要解决,假设用户输入的是一个合数,基于辛达拉姆筛的素数判定方法是否还优于常规判定方法。能够证明,无论输入的是合数还是素数,基于辛达拉姆筛的素数判定方法始终比常规方法快一倍。证明例如以下:

           假设用户输入的是素数,在分析复杂度的时候已经证明。假设用户输入的是合数,则常规方法在遇到a的最小质因子b时返回,而辛达拉姆判定方法会在遍历到第(b-1)/2行时返回。假设a=b*y,由于a是合数,则肯定存在p行和q列使得2pq+p+q=(a-1)/2。变换得(2p+1)(2q+1)=a。由于辛达拉姆判定方法在遍历时是从第一行開始的,所以我们能够觉得b=2p+1,从而辛达拉姆判定方法在第(b-1)/2行时返回。这也就证明了当输入是合数(奇数)时,基于辛达拉姆筛的素数判定方法也比常规方法快一倍。

    5 基于辛达拉姆筛的素数筛选

           辛达拉姆筛最初的目的和埃拉托色尼筛一样,也是筛选全部的素数,上面的素数判定算是一个变形应用,筛选全部素数才是筛的最重要目的。只是和埃氏筛法相比。辛达拉姆筛却不那么easy获得全部的素数。

    从辛达拉姆得到的结论:假设N出如今表中。则2N+1是合数。若N不在表中。则2N+1是素数,我们能够非常easy知道不在表中的全部数,其二倍加一包括了全部的奇素数。

    比如,表中出现的前10个数为4,7,10,12,13,16,17,19,22,24,不在表中的数即为1,2,3,5,6,8,9,11,14,15,18,20,21,23,这些数的二倍加一分别为:3,5,7,11,13,17,19,23,29,31,37,41,43,47,也即前14个奇素数。尽管我们仅仅须要找到全部不在表中的全部数就可以获得全部的奇素数,可是怎样找到却是个麻烦问题。

    问题在于怎样按序遍历在表中的全部数。本人临时没有思路,还请高手帮忙解答,在此谢过。

    參考

           对于辛达拉姆筛的了解是从高中一本数学习题集中获得的,当时读了练习后面的拓展阅读,印象很深刻。以至于到如今我都留着那本习题集,感谢它。


  • 相关阅读:
    CLRS2e读书笔记—Chapter10
    CLRS2e读书笔记—Chapter6
    CLRS2e读书笔记—Chapter34 & Appendix A,B
    CLRS2e读书笔记—Chapter8
    CLRS2e读书笔记—Chapter5 & Appendix C
    CLRS2e读书笔记—Chapter7
    Haskell学习笔记<三>
    屏幕一直亮的问题
    对CCLabelTTF进行自动换行,显示打字效果
    XCODE 4.2打包发布
  • 原文地址:https://www.cnblogs.com/mfrbuaa/p/5202129.html
Copyright © 2011-2022 走看看