即使我们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++");
}

本工程源码下载


04-14 13:17