根据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