前言
说到引用,对于一个Android开发工程师来说,肯定不会陌生,下面我们将介绍JNI中的引用。
引用类型
JNI中的引用分为局部引用和全局引用(包含弱全局引用)
引用的作用
在JNI中告知虚拟机何时回收一个JNI变量
局部引用
局部引用,通过DeleteLocalRef手动释放对象。
应用场景
1.访问一个很大的java对象,使用完之后,还要进行复杂的耗时操作
2.创建了大量的局部引用,占用了太多的内存,而且这些局部引用跟后面的操作没有关联性
具体例子
创建一个native方法
public native void localRef();
native函数的实现,模拟应用场景
JNIEXPORT void JNICALL
Java_com_zhangpan_myjnicmake_MainActivity_localRef(JNIEnv *env, jobject jobj) {
int i = 0;
for (; i < 5; i++) {
//创建Date对象
jclass jcla = (*env)->FindClass(env, "java/util/Date");
jmethodID jmid = (*env)->GetMethodID(env, jcla, "<init>", "()V");
jobject jobj = (*env)->NewObject(env, jcla, jmid);
//此处省略一百行代码...
//不再使用jobj对象了
//通知垃圾回收期回收这些对象
(*env)->DeleteLocalRef(env, jobj);
//此处省略一百行代码...
}
}
全局引用
共享(可以跨多个线程使用),手动控制内存使用
具体例子
定义三个native方法,分别为createGlobalRef()、getGlobalRef()、deleteGlobalRef()
public native void createGlobalRef();
public native String getGlobalRef();
public native void deleteGlobalRef();
在MainActivity中的onCreate中调用
createGlobalRef();
System.out.println(getGlobalRef());
deleteGlobalRef();
System.out.println("释放完了...");
//System.out.println(getGlobalRef()); //释放完了,再获取会报空指针
native函数的实现
jstring jstr;
JNIEXPORT void JNICALL
Java_com_zhangpan_myjnicmake_MainActivity_createGlobalRef(JNIEnv *env, jobject instance) {
jstring obj = (*env)->NewStringUTF(env, "China is Powerful");
jstr = (*env)->NewGlobalRef(env, obj);
}
JNIEXPORT jstring JNICALL
Java_com_zhangpan_myjnicmake_MainActivity_getGlobalRef(JNIEnv *env, jobject instance) {
return jstr;
}
JNIEXPORT void JNICALL
Java_com_zhangpan_myjnicmake_MainActivity_deleteGlobalRef(JNIEnv *env, jobject instance) {
(*env)->DeleteGlobalRef(env, jstr);
}
运行输出结果:
释放完了...
如果不相信已经将全局引用释放完了,我们可以在调用deleteGlobalRef之后,再调用getGlobalRef,运行之后会报空指针,这也就证明了在调用了deleteGlobalRef之后,全局引用也就释放了。
弱全局引用
创建弱全局引用:NewWeakGlobalRef,销毁全局引用:DeleteGlobalWeakRef
和全局引用不一样的是:
1.节省内存,在内存不足时可以释放所引用的对象
2.可以引用一个不常用的对象,如果为NULL(被回收了),可以手动临时再创建一个
项目地址:
https://github.com/fsrmeng/MyJniCmake-Master
展望
本篇博客介绍了JNI引用,下篇博客我将介绍JNI中的异常处理,敬请期待!
网友评论