【发布时间】:2019-01-06 16:31:25
【问题描述】:
如何提高 Java 的 Big Integer 的性能?
例如,这个阶乘程序:
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.5s 完成
在 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.0s 完成
和 Ruby(用于比较):
puts (2...99999).inject(:*)
在 4.4s (Ruby) 和 32.2s 在 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.6s 和0.7s 完成MulRange
编辑根据要求:
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.4s
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.1s 完成
编辑 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.7s 完成
EDIT 4 正如@Marco13 所说,最大的问题是字符串的创建,而不是 BigInteger 本身。
- 大整数:
3.0s - MutableBigInteger 破解:
10.1s - 字符串创建:~
20s
【问题讨论】:
-
这不是一个完全公平的比较;在 Java 中,您使用
BigInteger作为循环变量,在 C++ 中,您只是使用int。 -
^^ 解决方法是开始使用 int。并缓存 .valueOf ,否则您每次都会创建一个新的 BigInteger。
-
您可以尝试使用MutableBigInteger。
-
您已经消除了所有先前值的 GC 开销,但您仍然经常重新分配 int[]。我试图调整您的代码以预先分配一个 int[50,000] 但它似乎对我不起作用。我可能做错了。
-
在 java 8 中,编辑 2 运行 3.5 秒。比 C++ 慢 3 倍还不错
标签: java biginteger