非jni attach线程操作java对象

本文详细解析了在JNI环境中,JNIEnv指针在多线程间的不可传递性,并提供了正确的线程间JNIEnv指针生成与使用的方法。强调了JNIEnv指针仅在其关联线程中有效,不当使用将导致程序崩溃。介绍了如何通过JavaVM保存全局引用,以及在新线程中通过AttachCurrentThread生成新的JNIEnv指针,确保跨线程操作Java对象的安全。

参考博文

博文链接
JNI全局对象,及原生线程JNIENV传递https://blog.csdn.net/jiabailong/article/details/74940849
多线程中的JNIEnvhttps://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从线程解绑定,不然会造成野指针
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值