Как увеличить производительность большого целого Java?Увеличение производительности BigInteger Java
Например, эта программа вычисления факториала:
import java.math.*;
class Fac {
public static void main(String[] args) {
BigInteger i = BigInteger.ONE;
for(BigInteger z=BigInteger.valueOf(2);z.compareTo(BigInteger.valueOf(99999)) != 0;) {
i = i.multiply(z);
z = z.add(BigInteger.ONE);
}
System.out.println(i);
}
}
Эта программа завершена в 31.5
s
Где в C++:
#include <iostream>
#include <gmpxx.h>
using namespace std;
int main() {
mpz_class r;
r = 1;
for(int z=2;z<99999;++z) {
r *= mpz_class(z);
}
cout << r << endl;
}
завершено в 1.0
s
и Руби (для сравнения):
puts (2...99999).inject(:*)
завершено в 4.4
с (рубин) и 32.2
с в JRuby
А также Go (для сравнения):
package main
import (
"fmt"
"math/big"
)
func main() {
i := big.NewInt(1);
one := big.NewInt(1)
for z := big.NewInt(2); z.Cmp(big.NewInt(99999)) < 0; {
i.Mul(i,z);
z.Add(z,one)
}
fmt.Println(i);
}
завершена в 1.6
с и 0.7
с для MulRange
EDIT В соответствии с запросом:
import java.math.*;
class F2 {
public static void main(String[] args) {
BigInteger i = BigInteger.ONE, r = BigInteger.valueOf(2);
for(int z=2; z<99999 ; ++z) {
i = i.multiply(r);
r = r.add(BigInteger.ONE);
}
System.out.println(i);
}
}
продолжительность выполнения: 31.4
сек
EDIT 2 для тех, кто все еще думает, что первый и второй код Java несправедливо ..
import java.math.*;
class F3 {
public static void main(String[] args) {
BigInteger i = BigInteger.ONE;
for(int z=2; z<99999 ; ++z) {
i = i.multiply(BigInteger.valueOf(z));
}
System.out.println(i);
}
}
завершено в 31.1
s
EDIT 3 @OldCurmudgeon комментарий:
import java.math.*;
import java.lang.reflect.*;
class F4 {
public static void main(String[] args) {
try {
Constructor<?> Bignum = Class.forName("java.math.MutableBigInteger").getDeclaredConstructor(int.class);
Bignum.setAccessible(true);
Object i = Bignum.newInstance(1);
Method m = i.getClass().getDeclaredMethod("mul", new Class[] { int.class, i.getClass()});
m.setAccessible(true);
for(int z=2; z<99999 ; ++z) {
m.invoke(i, z, i);
}
System.out.println(i);
} catch(Exception e) { System.err.println(e); }
}
}
завершена в 23.7
сек
EDIT 4 Как заявил @ Marco13 самая большая проблема была в создании строки не на самом BigInteger ..
- BigInteger:
3.0
s - MutableBigInteger hack:
10.1
s - Создание строки: ~
20
сек
Это не вполне справедливое сравнение; в Java вы используете 'BigInteger' как переменную цикла, на C++ вы просто используете' int'. –
^^ Исправить было бы начать использовать int. И cache .valueOf, или вы будете создавать новый BigInteger каждый раз. –
Вы можете попробовать использовать [MutableBigInteger] (http://stackoverflow.com/a/8583188/823393). – OldCurmudgeon