NDK资源释放

来源:互联网 发布:防拷贝软件 编辑:程序博客网 时间:2024/05/29 01:55
JNI 中的 Local Reference 只在 native method 执行时存在,当 native method 执行完后自动失效。这种自动失效,使得对 Local Reference 的使用相对简单,native method 执行完后,它们所引用的 Java 对象的 reference count 会相应减 1。不会造成 Java Heap 中 Java 对象的内存泄漏。
而 Global Reference 对 Java 对象的引用一直有效,因此它们引用的 Java 对象会一直存在 Java Heap 中。程序员在使用 Global Reference时,需要仔细维护对 Global Reference 的使用。如果一定要使用 Global Reference,务必确保在不用的时候删除。就像在 C 语言中,调用malloc() 动态分配一块内存之后,调用 free() 释放一样。否则,Global Reference 引用的 Java 对象将永远停留在 Java Heap 中,造成 Java Heap 的内存泄漏。
 
1、什么需要释放? 


什么需要什么呢 ? JNI 基本数据类型是不需要释放的 , 如 jint , jlong , jchar 等等 。 我们需要释放是引用数据类型,当然也包括数组家族。如:jstring ,jobject ,jobjectArray,jintArray 等等。
当然,大家可能经常忽略掉的是 jclass ,jmethodID , 这些也是需要释放的
 
2、如何去释放?


1)      释放String
jstring jstr = NULL;
char* cstr = NULL;
//调用方法
jstr = (*jniEnv)->CallObjectMethod(jniEnv, mPerson, getName);
cstr = (char*) (*jniEnv)->GetStringUTFChars(jniEnv,jstr, 0);
__android_log_print(ANDROID_LOG_INFO, "JNIMsg", "getName  ---- >  %s",cstr );
//释放资源
(*jniEnv)->ReleaseStringUTFChars(jniEnv, jstr, cstr);
(*jniEnv)->DeleteLocalRef(jniEnv, jstr);
2)      释放 类 、对象、方法
(*jniEnv)->DeleteLocalRef(jniEnv, XXX);
 
“XXX” 代表 引用对象
3)      释放 数组家族
jobjectArray arrays = NULL;
 
jclass jclsStr = NULL;
 
jclsStr = (*jniEnv)->FindClass(jniEnv, "java/lang/String");
 
arrays = (*jniEnv)->NewObjectArray(jniEnv, len, jclsStr, 0);
 
(*jniEnv)->DeleteLocalRef(jniEnv, jclsStr);  //释放String类
 
(*jniEnv)->DeleteLocalRef(jniEnv, arrays); //释放jobjectArray数组
 
   
 
native method 调用 DeleteLocalRef() 释放某个 JNI Local Reference 时,首先通过指针 p 定位相应的 Local Reference 在 Local Ref 表中的位置,然后从 Local Ref 表中删除该 Local Reference,也就取消了对相应 Java 对象的引用(Ref count 减 1)。
原创粉丝点击