美文网首页
Android NDK(JNI)详解教程四 引用、释放资源

Android NDK(JNI)详解教程四 引用、释放资源

作者: Kael_Zhang的安卓笔记 | 来源:发表于2023-03-01 08:48 被阅读0次

JNI 引用

JNI 定义了八种 Java 基本类型,其余的 jobject、jclass、jarray、jxxxArray、jstring 等都是引用类型。JNI 的引用两个含义:Java 中的引用类型、C/C++ 中的指针,会带来的一个问题:如果引用被 JVM 释放了,C指针仍然指向一个地址,但对应的地址中数据已经被释放了!

  • 全局引用(GlobalReferences):全局有效。JVM 无法释放回收,必须通过调用 DeleteGlobalRef() 显式释放。
    创建全局引用:jobject NewGlobalRef(JNIEnv *env, jobject obj);
    释放全局引用:void DeleteGlobalRef(JNIEnv *env, jobject globalRef);
  • 弱全局引用(WeakGlobalReferences):全局有效,可以被 JVM 回收。
    创建弱全局引用:jobject NewWeakGlobalRef(JNIEnv *env, jobject obj);
    释放弱全局引用:void DeleteWeakGlobalRef(JNIEnv *env, jobject globalRef);
  • 局部引用(LocalReferences):在方法内创建,方法结束自动释放,但是如果消耗过多 JVM 资源,也可以手动释放,引用较多时建议使用完了就手动释放
    创建局部引用:jobject NewLocalRef(JNIEnv *env, jobject obj);
    释放局部引用:void DeleteLocalRef(JNIEnv *env, jobject globalRef);
    以下两种情况必须手动释放:
    1.引用一个很大的 Java 对象
    2.在 for 循环中创建了大量的引用。引用多了之后会报 ReferenceTable overflow 异常。
  • 问:哪些场景需要释放?
    答:JNI 函数内部创建的 jobject、jclass、jstring、jarray 等引用需要释放
创建 释放
FindClass DeleteLocalRef
NewString DeleteLocalRef
NewStringUTF DeleteLocalRef
NewObject DeleteLocalRef
NewXxxArray DeleteLocalRef
GetObjectField DeleteLocalRef
GetObjectClass DeleteLocalRef
GetObjectArrayElement DeleteLocalRef
  • 对于 GetStringChars、GetStringUTFChars、GetXxxArrayElements 基本类型数组,需要调用对应的 Release 方法去释放本地内存,这是必须的!

字段和方法 ID( jfieldID 、 jmethodID)

  • jfieldID 和 jmethodID 是常规的 C 指针类型,涉及到的函数有:GetFieldID / GetXxxField / SetXxxField、GetStaticFieldID / GetStaticXxxField / SetStaticXxxField、GetMethodID / CallXxxMethod、GetStaticMethodID / CallStaticXxxMethod
  • 在 C、C++ 中调用 java 对象的变量或者方法时常常会用到 jfieldID 和 jmethodID
#include <jni.h>
#include <string>

extern "C" JNIEXPORT jstring JNICALL
Java_com_sample_projectname_MainActivity_stringFromJNI(
        JNIEnv* env,
        jobject /* this */) {
    // get the class 
    jclass class_obj = (*env)->GetObjectClass(env, jobject );

    // get the field id, com_sample_projectname_MainActivity 类中有 age、name成员变量
    jfieldID id_age = (*env)->GetFieldID(env, class_obj , "age", "I");
    jfieldID id_name = (*env)->GetFieldID(env, class_obj , "name", "Ljava/lang/String;");

    // get the field value
    jint age = (*env)->GetIntField(env, jobject , id_age);
    jstring name = (*env)->GetIntField(env, jobject , id_name );

    std::string hello = "Hello from C++";
    return env->NewStringUTF(hello.c_str());
}

相关文章

网友评论

      本文标题:Android NDK(JNI)详解教程四 引用、释放资源

      本文链接:https://www.haomeiwen.com/subject/fnjlldtx.html