4. 【干货】火爆全网的《超全NDK精品教程》JNI 内存管理和优化,全局引用和局部引用 &static

根据Java的垃圾回收机制,只要有引用存在就不回触发该引用所指向Java对象的垃圾回收

根据Java的垃圾回收机制,只要有引用存在就不回触发该引用所指向Java对象的垃圾回收

c层创建java对象,不使用了, 需要怎么处理?

访问一个很大的java对象,使用完之后,还要进行复杂的耗时操作。

创建了大量的局部引用,占用了太多的内存,而且这些局部引用跟后面的操作没有关联性。

原因是: c++需要自己管理内存的! 

JNI的三种引用

JNI引用概念:引用变量。

引用类型:局部引用和全局引用(全局引用里面包含全局弱引用)。

作用:在JNI中告知虚拟机何时回收一个JNI变量。

引用类型:局部引用和全局引用(全局引用里面包含全局弱引用)。

1.局部引用

典型使用场景:

一个局部引用仅在创建它的native函数及该函数调用的函数中有效

访问一个很大的java对象,使用完之后,还要进行复杂的耗时操作。

创建了大量的局部引用,占用了太多的内存,而且这些局部引用跟后面的操作没有关联性。

局部引用,通过DeleteLocalRef手动释放对象。

JNIEXPORT jintArray JNICALL Java_com_haocai_jni_JniTest_localRef

(JNIEnv *env, jobject jobj, jint len) {

    int i = 0;

    for (; i < 5;i++) {

        //创建Date对象

        jclass cls = (*env)->FindClass(env, "java/util/Date");

        jmethodID constructor_mid = (*env)->GetMethodID(env, cls, "<init>", "()V");

        jobject obj = (*env)->NewObject(env, cls, constructor_mid);

        //此处省略大量代码

        //不再使用jobject对象了

        //通知垃圾回收器回收这些对象

        (*env)->DeleteLocalRef(env, obj);

    }

}

2.全局引用

主要作用:共享(可以跨多个线程),手动控制内存使用。

Java中提供三个方法,分别用于创建、获取、删除JNI全局引用。

创建:NewGlobalRef

获取: getGlobalRef

删除: DeleteGlobalRef

使用场景: java需要c层下载进度的回调, 需要把它全局化

//创建

JNIEXPORT void JNICALL Java_com_haocai_jni_JniTest_createGlobalRef

(JNIEnv *env, jobject jobj, jint len) {

    jstring obj = (*env)->NewStringUTF(env, "jni development is powerful!");

    global_str = (*env)->NewGlobalRef(env, obj);

}

//获取

JNIEXPORT jstring JNICALL Java_com_haocai_jni_JniTest_getGlobalRef

(JNIEnv *env, jobject jobj, jint len) {

    return global_str;

}

//释放

JNIEXPORT void JNICALL Java_com_haocai_jni_JniTest_deleteGlobalRef

(JNIEnv *env, jobject jobj, jint len) {

    (*env)->DeleteGlobalRef(env, global_str);

}

3.弱引用

和java中的弱引用一样

使用方法与全局引用类似:

通过NewWeakGlobalRef创建全局引用。

通过DeleteWeakGlobalRef删除全局引用。

与全局引用不一样的是,弱全局引用特点:

节省内存,在内存不足时可以是释放所引用的对象。

可以引用一个不常用的对象,如果为NULL的时候(被回收了),可以手动再临时创建一个。

与全局引用类似,弱引用可以跨方法、线程使用。但与全局引用很重要不同的一点是,弱引用不会阻止 GC 回收它引用的对象。所以在使用时需要多加小心,它所引用的对象可能是不存在的或者已经被回收。当本地代码中缓存的引用不一定要阻止 GC 回收它所指向的对象时,弱引用就是一个最好的选,

      我们在使用弱引用的时候,必须先判断这个弱引用是指向一个活动的对象,还是一个已经被GC回收来的对象。NULL引用指向jvm中的null对象,可以通过IsSameObject来判断这引用是否被回收。通过env->IsSameObject(clz, NULL)来判断,被回收了则返回 JNI_TRUE(或者 1),否则返回 JNI_FALSE)。

当我们的本地代码不再需要一个弱全局引用时,也应该调用 DeleteWeakGlobalRef 来释放它,如果不手动调用这个函数来释放所指向的对象,JVM 仍会回收弱引用所指向的对象,但弱引用本身在引用表中所占的内存永远也不会被回收。

使用举例:


if(JNI_FALSE==(*env)->IsSameObject(env,weakGlobalcls,NULL)){

//TODO 对象未被回收,可以使用

}else{

//TODO 对象被垃圾回收器回收,不能使用

}

JNI中几种引用的区别? 3种引用的区别

从JVM创建的对象传递到C/C++代码时会产生引用,由于Java的垃圾回收机制限制,只要对象有引用存在就不会被回收。所以无论在C/C++中还是Java中我们在使用引用的时候需要特别注意。下面讲下C/C++中的引用:

全局引用

全局引用可以跨多个线程,在多个函数中都有效。全局引用需要通过NewGlobalRef方法手动创建,对应的释放全局引用的方法为DeleteGlobalRef

局部引用

局部引用很常见,基本上通过JNI函数获取到的返回引用都算局部引用,局部引用只在单个函数中有效。局部引用会在函数返回时自动释放,当然我们也可以通过DeleteLocalRef方法手动释放。

弱引用

弱引用也需要自己手动创建,作用和全局引用的作用相似,不同点在于弱引用不会阻止垃圾回收器对引用所指对象的回收。我们可以通过NewWeakGlobalRef方法来创建弱引用,也可以通过DeleteWeakGlobalRef来释放对应的弱引用

3.  jni中 static的用处 

缓存策略 static, native层有一大堆方法需要去获取 name 属性 // 初始化全局静态缓存

局部静态缓存:即使方法被重复调用也不会反复的给static修饰的字段赋值。


JNIEXPORTvoidJNICALLJava_com_east_jni10_Sample_staticLocalCache

(JNIEnv*env,jclassj_clz,jstringvalue){

    // name 属性,赋值操作

    staticjfieldIDjfid=NULL;// 局部静态缓存,即使方法被重复调用也不会反复的去获取 jfieldID


    if(jfid){

        printf("fieldID is not null\n");

    }

    else{

        jfid=(*env)->GetStaticFieldID(env,j_clz,"name","Ljava/lang/String;");// 如果没有静态缓存 这个方法会被多次调用,会去反复的获取 jfieldID

    }

    (*env)->SetStaticObjectField(env,j_clz,jfid,value);

}

全局静态缓存:即使方法被重复调用也不会反复的给static修饰的字段赋值。


// 全局静态缓存

staticjfieldIDf_name_id=NULL;

staticjfieldIDf_name1_id=NULL;

staticjfieldIDf_name2_id=NULL;

JNIEXPORTvoidJNICALLJava_com_east_jni10_Sample_staticLocalCache

(JNIEnv*env,jclassj_clz,jstringvalue){

    // 因为是静态缓存,所以这个方法被反复调用,也不会反复的去获取 jfieldID

    (*env)->SetStaticObjectField(env,j_clz,f_name_id,value);

}

JNIEXPORTvoidJNICALLJava_com_east_jni10_Sample_initStaticCache

(JNIEnv*env,jclassj_clz){

    // 初始化全局静态缓存

    f_name_id=(*env)->GetStaticFieldID(env,j_clz,"name","Ljava/lang/String;");

    f_name1_id=(*env)->GetStaticFieldID(env,j_clz,"name1","Ljava/lang/String;");

    f_name2_id=(*env)->GetStaticFieldID(env,j_clz,"name2","Ljava/lang/String;");

}

https://juejin.cn/post/7040685348015308814

https://blog.csdn.net/q925092273/article/details/109293383?spm=1001.2014.3001.5502

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,029评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,395评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,570评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,535评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,650评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,850评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,006评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,747评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,207评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,536评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,683评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,342评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,964评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,772评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,004评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,401评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,566评论 2 349

推荐阅读更多精彩内容