How can I find the Square Root of a Java BigInteger?

后端 未结 19 1015
[愿得一人]
[愿得一人] 2020-11-28 08:10

Is there a library that will find the square root of a BigInteger? I want it computed offline - only once, and not inside any loop. So even computationally expensive solutio

相关标签:
19条回答
  • 2020-11-28 08:34

    I know of no library solution for your question. You'll have to import an external library solution from somewhere. What I give you below is less complicated than getting an external library.

    You can create your own external library solution in a class with two static methods as shown below and add that to your collection of external libraries. The methods don't need to be instance methods and so they are static and, conveniently, you don't have to instance the class to use them. The norm for integer square roots is a floor value (i.e. the largest integer less than or equal to the square root), so you may need only the one static method, the floor method, in the class below for the floor value and can choose to ignore the ceiling (i.e. the smallest integer greater than or equal to the square root) method version. Right now, they are in the default package, but you can add a package statement to put them in whatever package you find convenient.

    The methods are dirt simple and the iterations converge to the closest integer answer very, very fast. They throw an IllegalArgumentException if you try to give them a negative argument. You can change the exception to another one, but you must ensure that a negatve argument throws some kind of exception or at least doesn't attempt the computation. Integer square roots of negative numbers don't exist since we are not in the realm of imaginary numbers.

    These come from very well known simple iterative integer square root algorithms that have been used in hand computations for centuries. It works by averaging an overestimate and underestimate to converge to a better estimate. This may be repeated until the estimate is as close as is desired.

    They are based on y1 = ((x/y0) + y0) / 2 converging to the largest integer, yn, where yn * yn <= x.

    This will give you a floor value for a BigInteger square root, y, of x where y * y <= x and (y + 1) * (y + 1) > x.

    An adaptation can give you a ceiling value for BigInteger square root, y, of x where y * y >= x and (y - 1) * (y - 1) < x

    Both methods have been tested and work. They are here:

    import java.math.BigInteger;
    
    public class BigIntSqRoot {
    
    public static BigInteger bigIntSqRootFloor(BigInteger x)
            throws IllegalArgumentException {
        if (x.compareTo(BigInteger.ZERO) < 0) {
            throw new IllegalArgumentException("Negative argument.");
        }
        // square roots of 0 and 1 are trivial and
        // y == 0 will cause a divide-by-zero exception
        if (x .equals(BigInteger.ZERO) || x.equals(BigInteger.ONE)) {
            return x;
        } // end if
        BigInteger two = BigInteger.valueOf(2L);
        BigInteger y;
        // starting with y = x / 2 avoids magnitude issues with x squared
        for (y = x.divide(two);
                y.compareTo(x.divide(y)) > 0;
                y = ((x.divide(y)).add(y)).divide(two));
        return y;
    } // end bigIntSqRootFloor
    
    public static BigInteger bigIntSqRootCeil(BigInteger x)
            throws IllegalArgumentException {
        if (x.compareTo(BigInteger.ZERO) < 0) {
            throw new IllegalArgumentException("Negative argument.");
        }
        // square roots of 0 and 1 are trivial and
        // y == 0 will cause a divide-by-zero exception
        if (x == BigInteger.ZERO || x == BigInteger.ONE) {
            return x;
        } // end if
        BigInteger two = BigInteger.valueOf(2L);
        BigInteger y;
        // starting with y = x / 2 avoids magnitude issues with x squared
        for (y = x.divide(two);
                y.compareTo(x.divide(y)) > 0;
                y = ((x.divide(y)).add(y)).divide(two));
        if (x.compareTo(y.multiply(y)) == 0) {
            return y;
        } else {
            return y.add(BigInteger.ONE);
        }
    } // end bigIntSqRootCeil
    } // end class bigIntSqRoot
    
    0 讨论(0)
  • 2020-11-28 08:35

    The C# language has similar syntax to Java. I wrote this recursive solution.

        static BigInteger fsqrt(BigInteger n)
        {
            string sn = n.ToString();
            return guess(n, BigInteger.Parse(sn.Substring(0, sn.Length >> 1)), 0);          
        }
        static BigInteger guess(BigInteger n, BigInteger g, BigInteger last)
        {
            if (last >= g - 1 && last <= g + 1) return g;
            else return guess(n, (g + (n / g)) >> 1, g);
        }
    

    Call this code like this (in Java I guess it would be "System.out.print").

    Console.WriteLine(fsqrt(BigInteger.Parse("783648276815623658365871365876257862874628734627835648726")));
    

    And the answer is: 27993718524262253829858552106

    Disclaimer: I understand this method doesn't work for numbers less than 10; this is a BigInteger square root method.

    This is easily remedied. Change the first method to the following to give the recursive portion some room to breathe.

        static BigInteger fsqrt(BigInteger n)
        {
            if (n > 999)
            {
               string sn = n.ToString();
               return guess(n, BigInteger.Parse(sn.Substring(0, sn.Length >> 1)), 0);
            }
            else return guess(n, n >> 1, 0);            
        }
    
    0 讨论(0)
  • 2020-11-28 08:37
        BigDecimal BDtwo = new BigDecimal("2");
        BigDecimal BDtol = new BigDecimal(".000000001");    
    private BigDecimal bigIntSQRT(BigDecimal lNew, BigDecimal lOld, BigDecimal n) {
            lNew = lOld.add(n.divide(lOld, 9, BigDecimal.ROUND_FLOOR)).divide(BDtwo, 9, BigDecimal.ROUND_FLOOR);
            if (lOld.subtract(lNew).abs().compareTo(BDtol) == 1) {
                lNew = bigIntSQRT(lNew, lNew, n);
            }
        return lNew;
    }
    

    I was just working on this problem and successfully wrote a recursive square root finder in Java. You can change the BDtol to whatever you want, but this runs fairly quickly and gave me the follow example as a result:

    Original number 146783911423364576743092537299333563769268393112173908757133540102089006265925538868650825438150202201473025

    SQRT --> 383123885216472214589586756787577295328224028242477055.000000000

    Then for confirmation 146783911423364576743092537299333563769268393112173908757133540102089006265925538868650825438150202201473025.000000000000000000

    0 讨论(0)
  • 2020-11-28 08:37

    Here's a solution that does not use BigInteger.multiply or BigInteger.divide:

        private static final BigInteger ZERO  = BigInteger.ZERO;
        private static final BigInteger ONE   = BigInteger.ONE;
        private static final BigInteger TWO   = BigInteger.valueOf(2);
        private static final BigInteger THREE = BigInteger.valueOf(3);
    
        /**
         * This method computes sqrt(n) in O(n.bitLength()) time,
         * and computes it exactly. By "exactly", I mean it returns
         * not only the (floor of the) square root s, but also the
         * remainder r, such that r >= 0, n = s^2 + r, and
         * n < (s + 1)^2.
         *
         * @param n The argument n, as described above.
         *
         * @return An array of two values, where the first element
         *         of the array is s and the second is r, as
         *         described above.
         *
         * @throws IllegalArgumentException if n is not nonnegative.
         */
        public static BigInteger[] sqrt(BigInteger n) {
            if (n == null || n.signum() < 0) {
                throw new IllegalArgumentException();
            }
    
            int bl = n.bitLength();
            if ((bl & 1) != 0) {
                ++ bl;
            }
    
            BigInteger s = ZERO;
            BigInteger r = ZERO;
    
            while (bl >= 2) {
                s = s.shiftLeft(1);
    
                BigInteger crumb = n.testBit(-- bl)
                                    ? (n.testBit(-- bl) ? THREE : TWO)
                                    : (n.testBit(-- bl) ? ONE : ZERO);
                r = r.shiftLeft(2).add(crumb);
    
                BigInteger d = s.shiftLeft(1);
                if (d.compareTo(r) < 0) {
                    s = s.add(ONE);
                    r = r.subtract(d).subtract(ONE);
                }
            }
    
            assert r.signum() >= 0;
            assert n.equals(s.multiply(s).add(r));
            assert n.compareTo(s.add(ONE).multiply(s.add(ONE))) < 0;
    
            return new BigInteger[] {s, r};
        }
    
    0 讨论(0)
  • 2020-11-28 08:38

    This is an easy to understand way that may not have the best performance but it gives the solution for a single BigInteger in less than a second.

    public static BigInteger sqrt(BigInteger n) {
        BigInteger bottom = BigInteger.ONE;
        BigInteger top = n;
        BigInteger mid;
        while(true) {
            mid = top.add(bottom).divide(new BigInteger(""+2));
            top = mid;
            bottom = n.divide(top);
    //            System.out.println("top:    "+top);
    //            System.out.println("mid:    "+mid);
    //            System.out.println("bottom: "+bottom);
            if(top.equals(bottom)) {
                return top;
            }
        }
    }
    
    0 讨论(0)
  • 2020-11-28 08:39

    I can't verify the accuracy of them but there are several home grown solutions when googling. The best of them seemed to be this one: http://www.merriampark.com/bigsqrt.htm

    Also try the Apache commons Math project (once Apache recovers from its bombardment after the JCP blog post).

    0 讨论(0)
提交回复
热议问题