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

后端 未结 19 1013
[愿得一人]
[愿得一人] 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:16

    you can also use binary search to find the square root of x also you can multiply it to for example 10^10 and find an integer like m by binary search since m^2

    System.out.println(m.divide(10^5)+"."+m.mod(10^5));

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

    This is the best (and shortest) working solution I've found

    http://faruk.akgul.org/blog/javas-missing-algorithm-biginteger-sqrt/

    Here is the code:

      public static BigInteger sqrt(BigInteger n) {
        BigInteger a = BigInteger.ONE;
        BigInteger b = new BigInteger(n.shiftRight(5).add(new BigInteger("8")).toString());
        while(b.compareTo(a) >= 0) {
          BigInteger mid = new BigInteger(a.add(b).shiftRight(1).toString());
          if(mid.multiply(mid).compareTo(n) > 0) b = mid.subtract(BigInteger.ONE);
          else a = mid.add(BigInteger.ONE);
        }
        return a.subtract(BigInteger.ONE);
      }
    

    I've tested it and it's working correctly (and seems fast)

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

    I needed to have the square root for BigIntegers for implementing the quadratic sieve. I used some of the solutions here but the absolutely fastest and best solution so far is from Google Guava's BigInteger library.

    Documentation can be found here.

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

    As Jigar states, Newton's iteration is both quite simple to understand and to implement. I'll leave it up to others decide whether it is the most efficient algorithm or not for finding the square root of a number.

    With recursion it can be done in just about two lines.

    private static BigInteger newtonIteration(BigInteger n, BigInteger x0)
    {
        final BigInteger x1 = n.divide(x0).add(x0).shiftRight(1);
        return x0.equals(x1)||x0.equals(x1.subtract(BigInteger.ONE)) ? x0 : newtonIteration(n, x1);
    }
    

    Where n is the number we want to find the square root of, and x0 is the number from the previous call, which will always be 1 when initiate the first call from another method. So preferably, you will complement it with something like this as well;

    public static BigInteger sqrt(final BigInteger number)
    {
        if(number.signum() == -1)
            throw new ArithmeticException("We can only calculate the square root of positive numbers.");
        return newtonIteration(number, BigInteger.ONE);
    }
    
    0 讨论(0)
  • 2020-11-28 08:22

    Simplified Jim answer and improved performance.

    public class BigIntSqRoot {
        private static BigInteger two = BigInteger.valueOf(2L);
    
        public static BigInteger bigIntSqRootFloor(BigInteger x)
                throws IllegalArgumentException {
            if (checkTrivial(x)) {
                return x;
            }
            if (x.bitLength() < 64) { // Can be cast to long
                double sqrt = Math.sqrt(x.longValue());
                return BigInteger.valueOf(Math.round(sqrt));
            }
            // starting with y = x / 2 avoids magnitude issues with x squared
            BigInteger y = x.divide(two);
            BigInteger value = x.divide(y);
            while (y.compareTo(value) > 0) {
                y = value.add(y).divide(two);
                value = x.divide(y);
            }
            return y;
        }
    
        public static BigInteger bigIntSqRootCeil(BigInteger x)
                throws IllegalArgumentException {
            BigInteger y = bigIntSqRootFloor(x);
            if (x.compareTo(y.multiply(y)) == 0) {
                return y;
            }
            return y.add(BigInteger.ONE);
        }
    
        private static boolean checkTrivial(BigInteger x) {
            if (x == null) {
                throw new NullPointerException("x can't be null");
            }
            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 true;
            } // end if
            return false;
        }
    }
    
    0 讨论(0)
  • 2020-11-28 08:23

    Strange that nobody has mentioned it earlier but in java 9 you have sqrt in BigInteger, so you can just use it like that:

    BigInteger nine = BigInteger.valueOf(9);
    BigInteger three = nine.sqrt();
    

    https://docs.oracle.com/javase/9/docs/api/java/math/BigInteger.html#sqrt--


    EDIT-1

    Adding that there is another flavour of this function that, in addition to the floored square root, also returns the remainder.

    sqrtAndRemainder() BigInteger[]
    
    Returns an array of two BigIntegers containing the integer square root s
    of this and its remainder this - s*s, respectively.
    
    0 讨论(0)
提交回复
热议问题