登录 用户中心() [退出] 后台管理 注册
   
您的位置: 首页 >> SoftHub关联区 >> 主题: [android]ndk 中的巨坑与 NewGlobalRef     [回主站]     [分站链接]
[android]ndk 中的巨坑与 NewGlobalRef
clq
浏览(335) - 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);

    }
}





总数:1 页次:1/1 首页 尾页  
总数:1 页次:1/1 首页 尾页  


所在合集/目录



发表评论:
文本/html模式切换 插入图片 文本/html模式切换


附件:



NEWBT官方QQ群1: 276678893
可求档连环画,漫画;询问文本处理大师等软件使用技巧;求档softhub软件下载及使用技巧.
但不可"开车",严禁国家敏感话题,不可求档涉及版权的文档软件.
验证问题说明申请入群原因即可.

Copyright © 2005-2020 clq, All Rights Reserved
版权所有
桂ICP备15002303号-1