问题
I first wrote a code for Karatsuba algorithm using long. It works perfectly I think. Using the same logic I converted the code to BigInteger but for some reasons its giving StackOverflowError.
I cant figure out why. Please help.
EDIT1: The code for long also has a logical flaw. I am not sure what.
EDIT2: The code for long works now. I switched a "%" operator with "/" by mistake.
EDIT3: All good now. I changed .xor to .pow and == to .equals and fixed some bracket issues in the return statement. Thank you everyone for the help!
Here is the correct code:
public static BigInteger karatsuba3(BigInteger i, BigInteger j){
if (i.compareTo(Ten) == -1 || j.compareTo(Ten) == -1)
return i.multiply(j);
String length = getLength(i.max(j));
BigInteger n = new BigInteger(length);
if (n.mod(Two).equals(One))
n = n.add(One);
BigInteger a = i.divide(Ten.pow(n.divide(Two).intValue()));
BigInteger b = i.mod(Ten.pow(n.divide(Two).intValue()));
BigInteger c = j.divide(Ten.pow(n.divide(Two).intValue()));
BigInteger d = j.mod(Ten.pow(n.divide(Two).intValue()));
BigInteger first = karatsuba3(a,c);
BigInteger second = karatsuba3(b,d);
BigInteger third = karatsuba3(a.add(b),c.add(d));
return ((first.multiply(Ten.pow(n.intValue()))).add ((((third.subtract(first)).subtract( second))).multiply(Ten.pow(n.divide((new BigInteger("2"))).intValue()))).add(second));
}
Karatsuba with long code:
public static long karatsuba1(long i, long j){
if (i < 10 || j < 10)
return i*j;
double n = getLength(Math.max(i,j));
if (n%2 == 1)
n++;
long a = (long) (i/Math.pow(10,(n/2)));
long b = (long) (i%Math.pow(10,(n/2)));
long c = (long) (j/Math.pow(10,(n/2)));
long d = (long) (j%Math.pow(10,(n/2)));
long first = karatsuba1(a, c);
long second = karatsuba1(b, d);
long third = karatsuba1(a + b, c + d);
return ((long) ((first * Math.pow(10, n)) + ((third - first - second) * Math.pow(10, (n/2))) + second));
}
public static int getLength( long a){
String b = Long.toString(a);
return b.length();
}
Karatsuba with BigInteger code:
public static BigInteger karatsuba3(BigInteger i, BigInteger j){
BigInteger Ten = new BigInteger("10");
if (i.compareTo(Ten) == -1 || j.compareTo(Ten) == -1)
return i.multiply(j);
String length = getLength(i.max(j));
BigInteger n = new BigInteger(length);
if (n.mod(new BigInteger("2")) == new BigInteger("1"))
n.add(new BigInteger ("1"));
BigInteger a = i.divide(Ten.xor(n.divide((new BigInteger("2")))));
BigInteger b = i.mod(Ten.xor(n.divide((new BigInteger("2")))));
BigInteger c = j.divide(Ten.xor(n.divide((new BigInteger("2")))));
BigInteger d = j.mod(Ten.xor(n.divide((new BigInteger("2")))));
BigInteger first = karatsuba3(a,c);
BigInteger second = karatsuba3(b,d);
BigInteger third = karatsuba3(a.add(b),c.add(d));
return ((first.multiply(Ten.xor(n))).add (((third.subtract(first).subtract( second)))).multiply(Ten.xor(n.divide((new BigInteger("2"))))).add(second));
}
public static String getLength( BigInteger a){
String b = a.toString();
return Integer.toString(b.length());
}
回答1:
In the first snippet, this line looks wrong to me:
long d = (long) (j/Math.pow(10,(n/2)));
so right now you have c = d
probably you want:
long d = (long) (j%Math.pow(10,(n/2)));
回答2:
you have done some mistake when comparing and assigning bigint,so it is the cause for infinite recursive call.
if (n.mod(new BigInteger("2")) == new BigInteger("1"))
n.add(new BigInteger ("1"));
should be rectified as;
if (n.mod(new BigInteger("2")).equals(new BigInteger("1")))
n = n.add(new BigInteger ("1"));
but there are other issues that you should fix it to produce the same result as the "long" version. hope you will be able to rectify them.
回答3:
Use Ten.pow(x.intValue()) instead of Ten.xor(x)
回答4:
This works now
public static BigInteger karatsuba3(BigInteger x, BigInteger y) {
// cutoff to brute force
int N = Math.max(x.bitLength(), y.bitLength());
if (N <= 2000) return x.multiply(y); // optimize this parameter
// number of bits divided by 2, rounded up
N = (N / 2) + (N % 2);
// x = a + 2^N b, y = c + 2^N d
BigInteger b = x.shiftRight(N);
BigInteger a = x.subtract(b.shiftLeft(N));
BigInteger d = y.shiftRight(N);
BigInteger c = y.subtract(d.shiftLeft(N));
// compute sub-expressions
BigInteger ac = karatsuba3(a, c);
BigInteger bd = karatsuba3(b, d);
BigInteger abcd = karatsuba3(a.add(b), c.add(d));
return ac.add(abcd.subtract(ac).subtract(bd).shiftLeft(N)).add(bd.shiftLeft(2*N));
}
public static void main(String[] args){
System.out.println(karatsuba3(BigInteger.valueOf(12347634456346L), BigInteger.valueOf(23346345643565L)));
}
Found the solution here: Karatsuba Java
Now you can compare and see what's going wrong
来源:https://stackoverflow.com/questions/40124304/karatsuba-multiplication-using-biginteger