[android]ndk 中的巨坑与 NewGlobalRef
clq
浏览(566) -
2021-08-14 12:56:11 发表
编辑
关键字:
[2021-08-14 12:57:21 最后更新]
[android]ndk 中的巨坑与 NewGlobalRef
jobject context = View_getContext(obj);
java_ShowMessage("aaa", context); //奇怪,这里就是不行 //可看后面的直接代码,可以确信是某个指针无法保持引用造成的
//所以在 java 这样的自管理内存系统里指针的使用其实更是复杂。
//将涉及到的 class 全部改为全局引用就没有崩溃现象了。
return;
//---------------------------------------------------------------
//引用相关
//参考 onclick_t1 中的调用,当 class 这些的指针在 ndk 的多个事件中访问后是会失效的。这是因为 java 是自动管理内存的(所谓垃圾回收机制)
//其实这种现象在任何自已管理内存释放的系统中都会出现。只不过 ndk 的更不清晰。
//目前来说,已知的引用相关的函数、结构体如下:
//
// 参考 https://trent.blog.csdn.net/article/details/38365061
// myClass = (jclass)env->NewGlobalRef(tmp);
//
//
// 参考 https://blog.csdn.net/chuyouyinghe/article/details/46540091
// (*env)->DeleteLocalRef(env, _class); // 删除类指引
// (*env)->DeleteLocalRef(env,jthis); // 删除类对象指引
//
// 不过实测这种引用是会被收回的,应该不用自己删除.不过根据 https://blog.csdn.net/shulianghan/article/details/104193599
// 的说法。局部引用只是在其他地方不能使用而已,它的做法是立即转换成全局引用,然后立即删除本地引用。
//
//局部引用转换为全局引用 //函数里会删除局部引用,所以不用再手工删除了
//类和对象都转。不过查找到的函数指针应该是不用的
jclass change_class_ref(JNIEnv* env, jclass class_)
{
if (NULL == class_) return NULL;
jclass r = (*env)->NewGlobalRef(env, class_);
(*env)->DeleteLocalRef(env, class_);
return r;
}//
jobject change_obj_ref(JNIEnv* env, jobject obj)
{
if (NULL == obj) return NULL;
jobject r = (*env)->NewGlobalRef(env, obj);
(*env)->DeleteLocalRef(env, obj);
return r;
}//
//--------------------------------------------------------------- 引用 end;
guest
2021-08-14 12:57:21 发表
编辑
https://blog.csdn.net/shulianghan/article/details/104193599
【Android NDK 开发】JNI 引用 ( 全局引用 | NewGlobalRef | DeleteGlobalRef )
韩曙亮 2020-02-06 12:13:13 1375 收藏
分类专栏: # Android NDK 开发 文章标签: JNI 全局引用
版权
Android NDK 开发
专栏收录该内容
23 篇文章 5 订阅
订阅专栏
文章目录
I . 全局引用
II . 创建全局引用 ( NewGlobalRef )
III . 全局引用代码示例
I . 全局引用
1 . 全局引用作用域 :
与局部引用对比 : 全局引用与局部引用相对应 , 其作用域是全局的 , 局部引用只能在当前方法使用 ;
空间 : 可以 跨方法 , 跨线程使用 ;
时间 : 创建后可以使用 , 手动释放后全局引用失效 ; ( 手动释放前全局可用 )
2 . 全局引用 内存回收 : 全局引用 与 局部引用 均不会被 JVM 自动回收 , 如果内存不足 , JVM 宁可抛出 OOM 异常 , 也不会回收这些内存 ;
3 . 全局引用相关方法 :
① 创建全局引用 : NewGlobalRef ;
② 释放全局引用 : DeleteGlobalRef ;
II . 创建全局引用 ( NewGlobalRef )
1 . 函数原型 : 传入一个局部引用参数 , 将局部引用转为全局引用 ;
返回值 : 由局部引用转换成的全局引用 ;
参数 :
jobject obj : 使用 NewXXX / FindXXX 等 大多数 JNI 方法 默认创建的 Java 引用类型对象 都是局部引用 ;
struct _JNIEnv {
/* _JNIEnv 结构体中封装了 JNINativeInterface 结构体指针 */
const struct JNINativeInterface* functions;
...
// 最终 调用的 还是 JNINativeInterface 结构体中封装的 NewGlobalRef 方法
jobject NewGlobalRef(jobject obj)
{ return functions->NewGlobalRef(this, obj); }
...
}
1
2
3
4
5
6
7
8
9
III . 全局引用代码示例
全局引用代码示例:
// 全局引用
// 访问时如果局部变量也有同名变量 , 可以使用 域作用符 访问
// ::class_teacher 表示访问全局的变量
jclass class_teacher_global;
extern "C"
JNIEXPORT void JNICALL
Java_kim_hsl_jni_MainActivity_jniGlobalReferenceTest(JNIEnv *env, jobject instance) {
/*
全局引用 作用域 :
空间 : 可以 跨方法 , 跨线程使用
时间 : 创建后可以使用 , 手动释放后全局引用失效
全局引用创建 : NewGlobalRef
全局引用释放 : DeleteGlobalRef
全局引用会阻止 JVM 回收该引用
这里注意域作用符的使用 , 本方法中没有 class_teacher_global 同名变量 , :: 可用 可 不用
*/
// 1 . 获取 Teacher 类 ( 该变量需要释放 )
if(::class_teacher_global == NULL) {
//生成局部引用 , 该局部引用使用完毕后可释放
jclass tmp_class = env->FindClass("kim/hsl/jni/Teacher");
//将上述生成的局部引用变成 全局引用
// 全局引用释放时 , env->DeleteGlobalRef(class_teacher_global) 即可释放下面转换的 全局引用
::class_teacher_global = static_cast<jclass>(env->NewGlobalRef(tmp_class));
//将局部引用释放掉
env->DeleteLocalRef(tmp_class);
}
}
本帖子属于以下条目()
NEWBT官方QQ群1: 276678893
可求档连环画,漫画;询问文本处理大师等软件使用技巧;求档softhub软件下载及使用技巧.
但不可"开车",严禁国家敏感话题,不可求档涉及版权的文档软件.
验证问题说明申请入群原因即可.