最近天气变暖,阳光明媚,照在身上暖洋洋的人都变懒了,都不想都。但是作为程序猿,就没有享受的命,咱们继续接着来吧,这次主要说的是JNI的局部引用和全局引用还有JNI中的缓存的使用。那么开始吧。
我们在java中使用变量有JVM帮我们负责回收,不用自己手动干预,但是在JNI中我们创建的变量JVM是管理不了的,所以JNI提供了引用,实现类型JVM的功能。
JNI中的局部引用
env->NewLocalRef(object) 创建一个局部引用,在方法调用完毕后会自动释放。这个自动释放不代表我们就可以不用管了,使用局部引用是非常必要的。比如创建大对象,用完了,方法还没结束或者方法一直在循环,那么就要立即释放掉,不让它占用资源。再比如for循环中不停的创建对象,那么这个对象使用完之后也是要释放的,因为创建过多的对象可能造成引用表的溢出。所以我们要使用 env->DeleteLocalRef(object)来释放。局部引用不能跨线程调用,也不能跨方法调用
JNI中的全局引用
JNI中的全局引用类似与JAVA中的全局变量,定义过后可以跨方法使用,同时也能跨线程调用。
全局引用的创建 通过函数 env->NewGlobalRef(object)来创建 通过env->DeleteGlobalRef(object)来释放,代码如下
全局引用在不使用的情况下一定要释放
JNI中缓存的使用
JNI中缓存一般有两种,局部引用缓存和全局引用缓存,都是通过static关键字来实现的。这里只推荐用全局缓存,局部引用在方法结束之后就释放了,如果在别的方法中调用会报错。
全局缓存跟JAVA中的单例有点像,判断为空,空创建,不空直接使用。把上面的代码做个优化
我们在具体项目中使用的时候一般会在加载SO文件下面执行一个初始化程序init()提前加载和缓存要使用到的全局变量
上面的代码我们再优化一下
JAVA中代码初始化
JNI中全局弱引用
同全局引用使用方法差不多,但是不同点是在JVMGC内存不足的时候会释放引用的数据内存,但是不会释放这个引用,所以在用全局弱引用的时候,也一点要删除掉
这些引用实际上都是一些指针,在C/C++中每个指针都是会占用4个字节的内存的,所以对于弱全局引用,即时JVM会回收引用的对象,但是引用本事的指针是不会回收的,所以我们要调用DELE函数来释放。