参考博文
| 博文 | 链接 |
|---|---|
| JNI全局对象,及原生线程JNIENV传递 | https://blog.csdn.net/jiabailong/article/details/74940849 |
| 多线程中的JNIEnv | https://blog.csdn.net/Jefry2008/article/details/84097583 |
总结
在一个线程中,通过C++回调Java时,通过一个全局的env来操作java对象,这样程序会挂掉。文档中出现了这个很好的说明了这一点
A JNIEnv pointer is only valid in the thread associated with it. You must not
pass this pointer from one thread to another, or cache and use it in multiple
threads.
解决:
先通过另外一个含有JNIEnv 的线程中获取一个JavaVM *jvm;然后设外全局变量
然后在另外的一个线程中
JNIEnv *env;
jvm->AttachCurrentThread(&env, NULL);
在通过这个env操作Java对象
JNIENV *env 无法在多线程之间进行传递,这时就需要为原生线程生成新的JNIENV指针
需要先保存一个JavaVM的全局变量。如果需要保存Java全局对象则需要创建一个全局的jobject对象。
JavaVM *g_jvm = NULL;
jobject g_obj = NULL;
env->GetJavaVM(&g_jvm);
JNIENV*env
//Attach主线程
if (g_jvm->AttachCurrentThread(&env, NULL) != JNI_OK) {
LOGE("%s: AttachCurrentThread() failed", __FUNCTION__);
return NULL;
}
g_obj = env->NewGlobalRef(job);
env->DeleteLocalRef(g_obj);//使用完一定要释放,不然会造成内存泄漏
g_jvm->DetachCurrentThread();//一定要报JNIENV从线程解绑定,不然会造成野指针
本文详细解析了在JNI环境中,JNIEnv指针在多线程间的不可传递性,并提供了正确的线程间JNIEnv指针生成与使用的方法。强调了JNIEnv指针仅在其关联线程中有效,不当使用将导致程序崩溃。介绍了如何通过JavaVM保存全局引用,以及在新线程中通过AttachCurrentThread生成新的JNIEnv指针,确保跨线程操作Java对象的安全。
1728

被折叠的 条评论
为什么被折叠?



