大多数JNI函数返回局部引用,局部引用不能在后续的调用中被缓存及重用
一旦原生函数返回,局部引用立即被释放
例如FindClass函数返回一个局部引用,当原生方法返回时,它被自动释放,也可以用DeleteLocalRef函数显式释放原生代码
删除一个局部引用
jclass clazz;
clazz = (* env)->FindClass(env,"java/lang/String");
(*env)->DeleteLocalRef(env,clazz);
根据JNI规范,虚拟机应该允许原生代码创建最少16个局部引用
创建全局引用
可以用NewGlobalRef函数将局部引用初始化为全局引用
jclass localClazz;
jclass globalClazz;
localClazz = (* env)->FindClass(env,"java/lang/String");
globalClazz = (* env)->NewGlobalRef(env,localClazz);
//删除局部引用
(*env)->DeleteLocalRef(env,localClazz);
//删除全局引用
(*env)->DeleteGlobalRef(env,localClazz);
弱全局引用
全局引用的另一种类型的弱全局引用.
与全局引用一样,与全局引用一样在原生方法的 后续调用过程中依然有效
但弱全局引用并不阻止潜在的对象被垃圾回收.
创建弱全局引用
jclass localClazz;
jclass weakGlobalClazz;
localClazz = (* env)->FindClass(env,"java/lang/String");
weakGlobalClazz = (* env)->NewWeakGlobalRef(env,localClazz);
//弱全局引用的有效性校验
if(JNI_FALSE == (* env)->IsSameObject(env,weakGlobalClazz,NULL)){
//对象仍然处于活动状态且可以使用
}else{
//对象被垃圾回收器回收不能使用
}
//删除局部引用
(*env)->DeleteWeakGlobalRef(env,localClazz);