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());
}