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