Release<Type>ArrayElements 在 JNI 调用期间实际上没有从堆中释放内存?
Release<Type>ArrayElements not actually releasing memory from the heap during JNI call?
我正在开发一个 Android 应用程序,它与一些本机 C++ 代码 (JNI) 交互。在 Java 方面,我将查找 table (双数组)和两个 Open-CV 矩阵传递给 JNI(通过引用),然后使用 C++ 处理矩阵。虽然 JNI 函数调用在前 15~20 次有效,但应用程序很快就会崩溃并重新启动。我 99% 确定这是我没有正确释放堆内存的问题。
我查看了探查器以检查内存发生了什么,我发现每次 JNI 函数调用的内存使用量都在持续增加。大多数内存分配似乎都在本机部分,您可以在下图中看到这种增加(增加与对 JNI 函数的调用一致)。
extern "C" JNIEXPORT void JNICALL
Java_com_mygroup_productName_ImgProcUtils_interpVals(
JNIEnv *env,
jobject /* this */,
jlong addrKSqrd,
jint nRows,
jint nCols,
jdoubleArray yTaucVal,
jlong addrTauc) {
cv::Mat& kSqrd = *(cv::Mat*)addrKSqrd;
cv::Mat& Tauc = *(cv::Mat*)addrTauc;
jboolean isCopy;
jdouble *elem = env->GetDoubleArrayElements(yTaucVal, &isCopy);
float pixel;
for (int i = 0; i < nRows; i++) {
for (int j = 0; j < nCols; j++) {
pixel = kSqrd.at<float>(i, j);
int value = (int)round(pixel * 65535);
if (value < 0) {
value = 0;
} else if (value > 65535) {
value = 65535;
}
Tauc.at<float>(i,j) = (jfloat)elem[value];
}
}
env->ReleaseDoubleArrayElements(yTaucVal, elem, JNI_ABORT);
}
如您所见,我在 for 循环之前释放了双精度数组 I "get",但似乎我们还有未释放的内存。我是否需要做任何其他事情才能正确释放内存?我需要发布任何其他数据吗?
我怀疑是发布问题。
如果我有这样超级简单的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 进程的本机内存消耗(随时间推移)。
我正在开发一个 Android 应用程序,它与一些本机 C++ 代码 (JNI) 交互。在 Java 方面,我将查找 table (双数组)和两个 Open-CV 矩阵传递给 JNI(通过引用),然后使用 C++ 处理矩阵。虽然 JNI 函数调用在前 15~20 次有效,但应用程序很快就会崩溃并重新启动。我 99% 确定这是我没有正确释放堆内存的问题。
我查看了探查器以检查内存发生了什么,我发现每次 JNI 函数调用的内存使用量都在持续增加。大多数内存分配似乎都在本机部分,您可以在下图中看到这种增加(增加与对 JNI 函数的调用一致)。
extern "C" JNIEXPORT void JNICALL
Java_com_mygroup_productName_ImgProcUtils_interpVals(
JNIEnv *env,
jobject /* this */,
jlong addrKSqrd,
jint nRows,
jint nCols,
jdoubleArray yTaucVal,
jlong addrTauc) {
cv::Mat& kSqrd = *(cv::Mat*)addrKSqrd;
cv::Mat& Tauc = *(cv::Mat*)addrTauc;
jboolean isCopy;
jdouble *elem = env->GetDoubleArrayElements(yTaucVal, &isCopy);
float pixel;
for (int i = 0; i < nRows; i++) {
for (int j = 0; j < nCols; j++) {
pixel = kSqrd.at<float>(i, j);
int value = (int)round(pixel * 65535);
if (value < 0) {
value = 0;
} else if (value > 65535) {
value = 65535;
}
Tauc.at<float>(i,j) = (jfloat)elem[value];
}
}
env->ReleaseDoubleArrayElements(yTaucVal, elem, JNI_ABORT);
}
如您所见,我在 for 循环之前释放了双精度数组 I "get",但似乎我们还有未释放的内存。我是否需要做任何其他事情才能正确释放内存?我需要发布任何其他数据吗?
我怀疑是发布问题。
如果我有这样超级简单的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 进程的本机内存消耗(随时间推移)。