我有一个c ++库,我正在编写JNI包装器函数,允许我从JAVA调用它。我有很多不同的功能,但我特别遇到问题
我试图包装的功能如下:
void Set_Name(std::string invar)
我对GetStringUTFChars和ReleaseStringUTFChars进行了匹配调用,如下所示:
JNIEXPORT void JNICALL Java_com_metcarob_mys_javaapi_a_1SM_1D_1Orginisation_Set_1Name
(JNIEnv *p_jEnv, jobject p_jObj, jlong p_nat, jstring invar) {
SM_D_Orginisation* p_SMD = (SM_D_Orginisation*) p_nat;
const char *pPp_invar2 = p_jEnv->GetStringUTFChars( invar, NULL );
if (NULL==pPp_invar2) return;
std::string *pPp_invar = new std::string(pPp_invar2);
p_SMD->Set_Name(*pPp_invar);
pPp_invar2 = pPp_invar->c_str();
p_jEnv->ReleaseStringUTFChars(invar, pPp_invar2);
SAFE_DELETE(pPp_invar);
}
当我从Java运行时,我收到一个无效的指针异常:
*** glibc detected *** /home/robert/Oracle/Middleware/jdk160_21/bin/java: free(): invalid pointer: 0x0841a744 ***
======= Backtrace: =========
/lib32/libc.so.6(+0x6b511)[0xf7637511]
/lib32/libc.so.6(+0x6ce1b)[0xf7638e1b]
/lib32/libc.so.6(cfree+0x6d)[0xf763bf8d]
/home/robert/Oracle/Middleware/jdk160_21/jre/lib/i386/client/libjvm.so(+0x34fc8c)[0xf703bc8c]
/home/robert/Oracle/Middleware/jdk160_21/jre/lib/i386/client/libjvm.so(+0x25139a)[0xf6f3d39a]
/home/robert/Encrypted/Projects/Scout_Management/smbackend/DEBUG/libjavaapi.so(_ZN7JNIEnv_21ReleaseStringUTFCharsEP8_jstringPKc+0x27)[0xe02475c9]
/home/robert/Encrypted/Projects/Scout_Management/smbackend/DEBUG/libjavaapi.so(Java_com_metcarob_mys_javaapi_a_1SM_1D_1Orginisation_Set_1Name+0xd8)[0xe0245467]
[0xf4c0105d]
如果我删除对ReleaseStringURFChars的调用,它可以正常工作。文档声明我必须为GetStringUTFChars的每次调用调用ReleaseStringUTFChars以防止内存泄漏。
我不知道我做错了什么或我应该检查什么
有人可以帮忙吗?
由于
罗伯特
答案 0 :(得分:2)
首先使用
获取指针const char *pPp_invar2 = p_jEnv->GetStringUTFChars( invar, NULL );
然后你用
重新分配指针pPp_invar2 = pPp_invar->c_str();
首先必须释放内存,然后重新指定指针。
此外,您无需使用pPp_invar
分配new
,然后将其删除,只需要一个简单的p_SMD->Set_Name(pPp_invar2);
即可。