首页 > 解决方案 > 新建时JNI中C++代码中的数组如何释放数组的缓冲区?

问题描述

我目前正在使用 Java 作为 GUI 在 C++ 中编写一个小型仿真器。为了实现这一点,我从我的 C++ 代码中进行 JNI 调用,将数据数组传递给 GUI 应用程序。但是,由于我在测试运行中进行的调用量非常大,很明显在我传递数据的函数中发生了内存泄漏。

在我的程序运行之前: 在我的程序运行之前

在我的程序由于内存不足而运行并崩溃后:( 在此处输入图像描述 请忽略此程序当前使用的 CPU 使用率,我知道通过 JNI 重复调用效率低下,我有其他解决方法)

在对发生的事情进行彻底分析后,我得出结论,导致内存泄漏的不是 Java GUI 类,而是将数据数组传递给 Java GUI 的函数中的代码:


//java.env is the JNIEnv*
//setDisplay_ is a valid non-null jmethodID at runtime
//displayObject is a valid non-null jobject at runtime

void Display::setDisplay(vector<uint32_t>& a)
{
    jint* buffer = new jint[a.size()];
    for(int i = 0; i < a.size(); i++)
        buffer[i] = (jint)a[i];
    jintArray par = java.env->NewIntArray(a.size());
    java.env->SetIntArrayRegion(par, 0, a.size(), buffer);
    java.env->CallVoidMethod(displayObject, setDisplay_, par);
    //java.env->ReleaseIntArrayElements(par, buffer, 0);
    delete buffer;
}

我唯一能看到这个函数导致内存泄漏的是jintArray我完全不知道当它超出范围时会发生什么,所以我只能假设是我释放缓冲区时的问题。但是,查看其他使用带有数组的 JNI 的示例代码(例如:here),我注意到他们从不释放他们创建的数组。在挖掘JNI 文档时,我遇到了Release<NativeType>ArrayElements我认为是我正在寻找的方法,因为描述:

ReleaseArrayElements 例程 void ReleaseArrayElements(JNIEnv *env, ArrayType 数组, NativeType *elems, jint 模式); 通知 VM 本机代码不再需要访问 elems 的一系列函数。elems 参数是使用相应的 GetArrayElements() 函数从数组派生的指针。如有必要,此函数会将对 elems 所做的所有更改复制回原始数组。mode 参数提供有关如何释放数组缓冲区的信息。如果 elems 不是数组中元素的副本,则模式无效。否则,mode 会产生如下影响,如下表所示:

真正让我希望这是我特别需要的线是

mode 参数提供有关如何释放数组缓冲区的信息

然而,经过进一步检查,我不太确定这是我最初认为的方法,并且这已经在测试中证明了自己,并且它似乎会exit()导致失败(因为 JNI 如此臭名昭著)并且每次都会发生这种失败我使用文档中提供的任何模式运行它。

所以我真正的问题是:New<PrimitiveType>Array在 JNI 中从 C++ 代码创建时,如何释放<PrimitiveType>Array's 缓冲区?

标签: javac++java-native-interfacejnienv

解决方案


经过一番挖掘后,我发现我需要在使用 NewIntArray 创建的数组上调用 ReleaseIntArrayElements 吗?@gerbit 的简短回答:

您只需要发布参考:

jintArray pixels = env->NewIntArray(width * height);
env->DeleteLocalRef(pixels)

所以很明显,当在 Java 调用 C++ 的方向上使用 JNI 时,你不需要清理你<PrimitiveType>Array的 's,因为 java 会为你处理这个。但是从C++方向调用Java时,DeleteLocalRef()为了防止内存泄漏,需要调用。


推荐阅读