1 回答
TA贡献1834条经验 获得超8个赞
我怀疑是释放问题。
如果我有像这样的超级简单的Java代码
package recipeNo026;
public class PassArray {
public static native void passDoubleArray(double[] array);
static { System.loadLibrary("PassArray"); }
public static void main(String[] args) throws Exception {
for(int i=0; i<100; i++) {
double[] doubleArray = new double[1_000_000_000];
passDoubleArray(doubleArray);
Thread.sleep(1000);
}
}
}
JNI本机代码除了调用stuff之外没有其他调用
JNIEXPORT void JNICALL Java_recipeNo026_PassArray_passDoubleArray
(JNIEnv * env, jclass obj, jdoubleArray array) {
printf ("Double array\n");
jboolean isCopy;
jdouble *doubleBody = (*env)->GetDoubleArrayElements(env, array, &isCopy);
(*env)->ReleaseDoubleArrayElements(env, array, doubleBody, JNI_ABORT);
}
Java 堆和本机代码的内存消耗似乎都非常稳定。您可以看到代码继续运行时本机内存是如何分配和释放的。
我肯定会开始寻找从您的包装器调用的部分代码中的泄漏JNI。
另外,请注意这样一个事实,即使您根本不调用本机代码(例如JNI),本机内存也会增长。毕竟,Java 在某些时候必须使用malloc它自己的堆进行分配。看看这里:
public static void main(String[] args) throws Exception {
int size = 10;
double [][] array = new double[100][1];
for(int i=0; i<100; i++) {
array[i] = new double[size];
size = size * 2;
System.out.println("Allocating: " + size);
Thread.sleep(1000);
}
}
没有JNI电话了。现在,让我们运行该应用程序。
> java -Xmx4G -Xms512m -Djava.library.path=:./lib -cp target recipeNo026.PassArray
library: :./lib
Allocating: 20
Allocating: 40
Allocating: 80
Allocating: 160
Allocating: 320
Allocating: 640
Allocating: 1280
Allocating: 2560
Allocating: 5120
Allocating: 10240
Allocating: 20480
Allocating: 40960
Allocating: 81920
Allocating: 163840
Allocating: 327680
Allocating: 655360
Allocating: 1310720
Allocating: 2621440
Allocating: 5242880
Allocating: 10485760
Allocating: 20971520
Allocating: 41943040
Allocating: 83886080
Allocating: 167772160
Allocating: 335544320
Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
at recipeNo026.PassArray.main(PassArray.java:25)
让我们看一下 Java 进程的本机内存消耗(随着时间的推移)。
添加回答
举报