我目前正在使用 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 缓冲区?
萧十郎
相关分类