即使我们Java层的函数没有参数,原生方法还是自带了两个参数,其中第一个参数就是JNIEnv。如下:
native方法:
public native String stringFromC(); public native String stringFromCpp();
原生方法:
jstring Java_com_example_jni_MainActivity_stringFromC(JNIEnv* env,jobject thiz){ return (*env)->NewStringUTF(env,"I am from C"); } extern "C" jstring Java_com_example_jni_MainActivity_stringFromCpp(JNIEnv* env,jobject thiz){ return env->NewStringUTF("I am from C++"); }
JNIEnv是指向可用JNI函数表的接口指针,原生代码通过JNIEnv接口指针提供的各种函数来使用虚拟机的功能。JNIEnv是一个指向线程-局部数据的指针,而线程-局部数据中包含指向线程表的指针。实现原生方法的函数将JNIEnv接口指针作为它们的第一个参数。
原生代码是C以及原生代码是C++其调用JNI函数的语法不同,C代码中,JNIEnv是指向JNINativeInterface结构的指针,为了访问任何一个JNI函数,该指针需要首先被解引用。因为C代码中的JNI函数不了解当前的JNI环境,JNIEnv实例应该作为第一个参数传递给每一个JNI函数调用者。
正确的写法应该是下面这样:
jstring Java_com_example_jni_MainActivity_stringFromC(JNIEnv* env,jobject thiz){ return (*env)->NewStringUTF(env,"I am from C"); }
然而,在C++代码中,JNIEnv实际上是C++类实例,JNI函数以成员函数形式存在,因为JNI方法已经访问了当前的JNI环境,因此JNI方法调用不要求JNIEnv实例作参数,在C++中,完成同样的功能代码应该是下面这样:
extern "C" jstring Java_com_example_jni_MainActivity_stringFromCpp(JNIEnv* env,jobject thiz){ return env->NewStringUTF("I am from C++"); }
本工程源码下载