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 进程的本机内存消耗(随时间推移)。