Warning: file_get_contents(/data/phpspider/zhask/data//catemap/9/java/318.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Java 米勒-拉宾试验:不可能的结果_Java_Debugging_Primality Test - Fatal编程技术网

Java 米勒-拉宾试验:不可能的结果

Java 米勒-拉宾试验:不可能的结果,java,debugging,primality-test,Java,Debugging,Primality Test,我试图在Java上实现Miller-Rabin素性测试,并用BigInteger类的本机素性测试来应对其计算时间。既然我在这里,你可能已经猜到我的代码不起作用了。问题是,我得到的错误是一个数学女士说不可能的错误。我想知道我做错了什么 米勒-拉宾素性测试背后的思想,没有太多的数学计算,是所有素数都满足一个适当性。如果不满足该适当性,则该数字为复合数;然而,如果适当性得到满足,则该数可以是素数,也可以属于称为伪素数的复合数的子集。绝对不可能发生的是测试将质数识别为复合数。 这正是运行我的代码时发生的

我试图在Java上实现Miller-Rabin素性测试,并用
BigInteger
类的本机素性测试来应对其计算时间。既然我在这里,你可能已经猜到我的代码不起作用了。问题是,我得到的错误是一个数学女士说不可能的错误。我想知道我做错了什么

米勒-拉宾素性测试背后的思想,没有太多的数学计算,是所有素数都满足一个适当性。如果不满足该适当性,则该数字为复合数;然而,如果适当性得到满足,则该数可以是素数,也可以属于称为伪素数的复合数的子集。绝对不可能发生的是测试将质数识别为复合数。
这正是运行我的代码时发生的事情

我在我的代码中搜索数学错误,我想没有。我尝试搜索Java错误,但没有发现任何错误。显然,有一些东西(或许多东西)我没有看到,所以我想寻求帮助


下面是一个静态类的
main
,该类创建的目的是为了容纳Miller-Rabin测试的实现,它在
main
之后显示。它不是概率测试,而是确定性测试:该方法搜索所有可能的见证人,如果找到,则返回
false
(即非素数);否则返回
true

    public static void main(String[] args) {
        long start;
        Random random = new Random();
        int N = Integer.parseInt("10");
        BigInteger a,b,c,d;

        long stopMR, stopNative;
        boolean answerMR, answerNative;

        for(int i=0 ; i<6 ; i++, N*=3)
            {
            a = new BigInteger(N, random);
            System.out.printf("\tTEST %d\n\nThe number to be checked is: \n\t %s\n"     + 
    //              "Written in 2-base:  \n\t %s\n"                     +
                    "Number of bits of a: %d \n", i,
                    a.toString(), 
    //              a.toString(2), 
                    a.toString(2).length());

            start = System.nanoTime();
            answerMR = primalityTest_MillerRabin(a);
            stopMR = System.nanoTime();
            stopMR -= start;

            start = System.nanoTime();
            answerNative = a.isProbablePrime(40);
            stopNative = System.nanoTime();
            stopNative -= start;

            System.out.printf("The test of Miller-Rabin said that the number %s.\n"
                    + "The native test said that the number %s.\n"
                    + "The time of MR is %d.\n"
                    + "The time of the native is %d.\n"
                    + "The difference Time(MR)-Time(native) is %d.\n",
                    answerMR        ? "is prime" : "isn't prime"   ,
                    answerNative    ? "is prime" : "isn't prime"   ,
                    stopMR, stopNative, stopMR - stopNative
                    );

            a = BigInteger.probablePrime(N, random);
            System.out.printf("\tTEST %d\n\nThe number to be checked is: \n\t %s\n"     + 
    //              "Written in 2-base:  \n\t %s\n"                     +
                    "Number of bits of a: %d \n", i,
                    a.toString(), 
    //              a.toString(2), 
                    a.toString(2).length());

            start = System.nanoTime();
            answerMR = primalityTest_MillerRabin(a);
            stopMR = System.nanoTime();
            stopMR -= start;

            start = System.nanoTime();
            answerNative = a.isProbablePrime(40);
            stopNative = System.nanoTime();
            stopNative -= start;

            System.out.printf("The test of Miller-Rabin said that the number %s.\n"
                    + "The native test said that the number %s.\n"
                    + "The time of MR is %d.\n"
                    + "The time of the native is %d.\n"
                    + "The difference Time(MR)-Time(native) is %d.\n=====\n",
                    answerMR        ? "is prime" : "isn't prime"   ,
                    answerNative    ? "is prime" : "isn't prime"   ,
                    stopMR, stopNative, stopMR - stopNative
                    );
            }

    }

    /** Tests {@code n} for primality using the <i>Miller-Rabin algorithm</i>.
     * 
     * <p><br> For {@code n} minor than <b>3,825,123,056,546,413,051</b> (i.e. roughtly four millions of millions of millions,
     *  i.e. 4·10<sup>18</sup>) the test is deterministic and have time complexity of Θ<font size=+1>(</font>10·modPow(·,n)<font size=+1>)</font>.
     * <br> For {@code n} greater than <b>3,825,123,056,546,413,051</b> the test is deterministic and have time complexity of 
     * Θ<font size=+1>(</font>2·log<sub>2</sub><sup>2</sup>(n)·modPow(·,n)<font size=+1>)</font>.
     * 
     * @param n
     * @return
     */
    public static boolean primalityTest_MillerRabin(BigInteger n){
        // If n is divided by 2 or is less than 2, then n is not prime.
        if( n.intValue()%2== 0       ||       n.equals(TWO) )
            {
            System.out.printf("The number is even.\n");
            return false;
            }

        // n = d*2^s +1
        BigInteger pMinus1 = n.subtract(ONE);

        int s = 0;
        while (pMinus1.mod(TWO).equals(ZERO)) 
            {
            s++;
            pMinus1 = pMinus1.divide(TWO);
            }
        BigInteger d = pMinus1;

        System.out.printf("%s is %s*2^%d+1.\n", n.toString(), d.toString(),s);

        // Old code:
        // pMinus1.divide(BigInteger.valueOf(2L << r - 1));


        // For some n is known what witness one has to choose in order to verify is n is composite.
        // While the code for EVERY known limit is shown, only those not-redundant is not comment.

        if(n.compareTo(LIMIT_2047)<0)
            return  ! isTWOWitnessOfCompositenessOfN_MR(                     n, d, s)           ;
        if(n.compareTo(LIMIT_9080191)<0)
            return  ! (
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(31) , n, d, s)          ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(73) , n, d, s)          );
        if(n.compareTo(LIMIT_4759123141)<0)
            return  ! (
                    isTWOWitnessOfCompositenessOfN_MR(                       n, d, s)           ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(7) , n, d, s)           ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(61) , n, d, s)          );


        if(n.compareTo(LIMIT_1122004669633)<0)
            return  ! (
                    isTWOWitnessOfCompositenessOfN_MR(                        n, d, s)          ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(13) , n, d, s)          ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(23) , n, d, s)          ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(1662803) , n, d, s) );
        if(n.compareTo(LIMIT_2152302898747)<0)
            return  ! (
                    isTWOWitnessOfCompositenessOfN_MR(                       n, d, s)           ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(3) , n, d, s)           ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(5) , n, d, s)           ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(7) , n, d, s)           ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(11) , n, d, s)          );
        if(n.compareTo(LIMIT_3474749660383)<0)
            return  ! (
                    isTWOWitnessOfCompositenessOfN_MR(                       n, d, s)           ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(3) , n, d, s)           ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(5) , n, d, s)           ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(7) , n, d, s)           ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(11) , n, d, s)          ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(13) , n, d, s)          );
        if(n.compareTo(LIMIT_341550071728321)<0)
            return  ! (
                    isTWOWitnessOfCompositenessOfN_MR(                       n, d, s)           ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(3) , n, d, s)           ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(5) , n, d, s)           ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(7) , n, d, s)           ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(11) , n, d, s)          ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(13) , n, d, s)          ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(17) , n, d, s)          );
        if(n.compareTo(LIMIT_3825123056546413051)<0)
            return  ! (
                    isTWOWitnessOfCompositenessOfN_MR(                       n, d, s)           ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(3) , n, d, s)           ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(5) , n, d, s)           ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(7) , n, d, s)           ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(11) , n, d, s)          ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(13) , n, d, s)          ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(17) , n, d, s)          ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(19) , n, d, s)          ||
                    isAWitnessOfCompositenessOfN_MR( BigInteger.valueOf(23) , n, d, s)          );


        // ...otherwise the program does an exaustive search for witness
        System.out.printf("The Miller-Rabin Test has no shortcuts.\n");


        boolean witnessFound = false;
        int logn = (int) log(n,2) +1;
        BigInteger limit = (    n.subtract(ONE) ).min(  BigInteger.valueOf(2*logn*logn)   );

        for(BigInteger a = TWO ; witnessFound && a.compareTo(limit)<=0 ; a.add(ONE))
                witnessFound = isAWitnessOfCompositenessOfN_MR( a , n, d, s);

        return !witnessFound;
    }

    /** Return {@code true} if and only if {@code a} is a witness for the compositeness of {@code n}, i.e. if and only if: 
     * <ol> n = d·2<sup>s</sup> + 1                         <br>
     *      gcd(a,n) = 1    (i.e. a doesn't divide n)       <br>
     *      _<br>
     *      a<sup>d</sup> ≠ 1 (mod n)                       <br>
     *      a<sup>(d·2^r)</sup> ≠ -1 (mod n)        for all rϵ[0,s)         </ol>
     * 
     * If the method returns {@code true} then {@code n} is definitely a composite number. However if the method returns {@code false} it 
     * still may be possible for {@code n} to be composite.
     * 
     * <p>If this method recognize {@code a} as a witness for the compositeness of {@code n}
     * 
     * @param a
     * @param n
     * @param d
     * @param s
     * @return
     */
    public static boolean isAWitnessOfCompositenessOfN_MR(BigInteger a, BigInteger n, BigInteger d, int s){
        System.out.printf("\t Verifying if %s is a witness of the compositness of %s.\n", a.toString(), n.toString());
        if( a.gcd(n) == ONE )
            {
            BigInteger dMultiplyTwoPowR = a.modPow(d, n),
                        nMinusOne = NEGATIVE_ONE.mod(n);
            boolean answer = dMultiplyTwoPowR != ONE;
            for(int r=1 ; answer && r<s ; r++)
                {
                System.out.printf("\t\t Testing r=%d.\n", r);
                    answer = answer && 
                            dMultiplyTwoPowR.modPow(TWO, n) != nMinusOne;
                }

            System.out.printf("\t The number %s %s a witness of the compositness of %s.\n", a.toString(), answer ? "is" : "isn't", n.toString());
            return answer;
            }
        else 
            {
            System.out.printf("\t %s divides %s.\n", a.toString(), n.toString());
            return true;
            }
    }

        /** Returns {@code isAWitnessOfCompositenessOfN_MR(TWO, n, d, s)}. 
         * 
         * <p><u><b>Warning:</b></u> This method avoids to control if gcd(2, {@code n})=1.
             * 
             * @param n
             * @param d
         * @param s
         * @return
         */
    public static boolean isTWOWitnessOfCompositenessOfN_MR( BigInteger n, BigInteger d, int s){
        System.out.printf("\t Verifying if 2 is a witness of the compositness of %s.\n", n.toString());
        BigInteger dMultiplyTwoPowR = TWO.modPow(d, n),
                nMinusOne = NEGATIVE_ONE.mod(n);
        boolean answer = dMultiplyTwoPowR != ONE;
        for(int r=1 ; answer && r<s ; r++)
             {
            System.out.printf("\t\t Testing r=%d.\n", r);
            answer = answer && 
                    dMultiplyTwoPowR.modPow(TWO, n) != nMinusOne;
             }

        System.out.printf("\t The number 2 %s a witness of the compositness of %s.\n", answer ? "is" : "isn't", n.toString());
        return answer;
    }

有几个表达式使用==和!=比较BigInteger对象。这必须由对equals的调用来替换,从而对!=”的结果求反

此外,我认为N_MR的ISAwitness of Composite中存在复制粘贴错误:

! dMultiplyTwoPowR.modPow(TWO, n).equals( nMinusOne );
我认为两个应该由a代替

编辑日志中的错误。使用以下代码:

public static double log(BigInteger x, base b){
    String support = x.toString();
    int n = support.length();
    double log10 = n + Math.log10( Double.parseDouble("0."+support) );
    return log10/Math.log10(b);
}

我想会有更多的错误,但是这些修复应该会有所帮助。

哪个素数失败了?log(biginger,int)来自哪里?@iamnotmaynard:例如,数字1021和954751201是通过方法
biginger.probablePrime()生成的
但是对于这个米勒·拉宾来说,数字
2
是他们组合性的见证。@laune:我现在将代码添加到原来的帖子中。
public static double log(BigInteger x, base b){
    String support = x.toString();
    int n = support.length();
    double log10 = n + Math.log10( Double.parseDouble("0."+support) );
    return log10/Math.log10(b);
}