Android NDK(JNI)詳解教程四 引用、釋放資源

JNI 引用

JNI 定義了八種 Java 基本類型,其余的 jobject、jclass、jarray、jxxxArray、jstring 等都是引用類型。JNI 的引用兩個含義:Java 中的引用類型、C/C++ 中的指針,會帶來的一個問題:如果引用被 JVM 釋放了,C指針仍然指向一個地址,但對應(yīng)的地址中數(shù)據(jù)已經(jīng)被釋放了!

  • 全局引用(GlobalReferences):全局有效。JVM 無法釋放回收,必須通過調(diào)用 DeleteGlobalRef() 顯式釋放。
    創(chuàng)建全局引用:jobject NewGlobalRef(JNIEnv *env, jobject obj);
    釋放全局引用:void DeleteGlobalRef(JNIEnv *env, jobject globalRef);
  • 弱全局引用(WeakGlobalReferences):全局有效,可以被 JVM 回收。
    創(chuàng)建弱全局引用:jobject NewWeakGlobalRef(JNIEnv *env, jobject obj);
    釋放弱全局引用:void DeleteWeakGlobalRef(JNIEnv *env, jobject globalRef);
  • 局部引用(LocalReferences):在方法內(nèi)創(chuàng)建,方法結(jié)束自動釋放,但是如果消耗過多 JVM 資源,也可以手動釋放,引用較多時建議使用完了就手動釋放
    創(chuàng)建局部引用:jobject NewLocalRef(JNIEnv *env, jobject obj);
    釋放局部引用:void DeleteLocalRef(JNIEnv *env, jobject globalRef);
    以下兩種情況必須手動釋放:
    1.引用一個很大的 Java 對象
    2.在 for 循環(huán)中創(chuàng)建了大量的引用。引用多了之后會報 ReferenceTable overflow 異常。
  • 問:哪些場景需要釋放?
    答:JNI 函數(shù)內(nèi)部創(chuàng)建的 jobject、jclass、jstring、jarray 等引用需要釋放
創(chuàng)建 釋放
FindClass DeleteLocalRef
NewString DeleteLocalRef
NewStringUTF DeleteLocalRef
NewObject DeleteLocalRef
NewXxxArray DeleteLocalRef
GetObjectField DeleteLocalRef
GetObjectClass DeleteLocalRef
GetObjectArrayElement DeleteLocalRef
  • 對于 GetStringChars、GetStringUTFChars、GetXxxArrayElements 基本類型數(shù)組,需要調(diào)用對應(yīng)的 Release 方法去釋放本地內(nèi)存,這是必須的!

字段和方法 ID( jfieldID 、 jmethodID)

  • jfieldID 和 jmethodID 是常規(guī)的 C 指針類型,涉及到的函數(shù)有:GetFieldID / GetXxxField / SetXxxField、GetStaticFieldID / GetStaticXxxField / SetStaticXxxField、GetMethodID / CallXxxMethod、GetStaticMethodID / CallStaticXxxMethod
  • 在 C、C++ 中調(diào)用 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());
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容