一、 一个疑问
在进行JNI编程中,同样一个函数FindClass
,C和C++中有不同的用法:
如果是C++要用 env->FindClass(str)
如果是C要用 (*env)->FindClass(env, str)
类似的区别几乎涉及到每一个结构体函数的使用,这是为什么呢?
二、问题分析
分析JNI的头文件jni.h
1. 头文件位置
libnativehelper/include_jni/jni.h (Android9.0)
libnativehelper/include/nativehelpe/jni.h (Android4.4)
2. JNIEnv的定义
代码定义:
#if defined(__cplusplus)
typedef _JNIEnv JNIEnv;
typedef _JavaVM JavaVM;
#else
typedef const struct JNINativeInterface* JNIEnv;
typedef const struct JNIInvokeInterface* JavaVM;
#endif
由此可见,在C和C++环境中的JNIEnv定义是不同的。
3. C环境中的JNIEnv
由定义处可见,C环境中JNIEnv实际上是一个JNINativeInterface*类型的结构体指针。
JNINativeInterface结构体代码定义:
/*
* Table of interface function pointers.
*/
struct JNINativeInterface {
...........................................
jint (*GetVersion)(JNIEnv *);
jclass (*DefineClass)(JNIEnv*, const char*, jobject, const jbyte*,
jsize);
jclass (*FindClass)(JNIEnv*, const char*);
...........................................
};
C中定义的JNIEnv* env实际上是JNINativeInterface** env类型的二维指针,因此在使用FindClass函数时,要将二维指针降为一维指针进行引用,并且根据定义需要传入二维的JNINativeInterface** 指针(env),即 (*env)->FindClass(env, str);
4. C++环境中的JNIEnv
由定义处可见,C++中的JNIEnv实际上是一个_JNIEnv类型的结构体。
_JNIEnv结构体代码定义:
/*
* C++ object wrapper.
*
* This is usually overlaid on a C struct whose first element is a
* JNINativeInterface*. We rely somewhat on compiler behavior.
*/
struct _JNIEnv {
/* do not rename this; it does not seem to be entirely opaque */
const struct JNINativeInterface* functions;
#if defined(__cplusplus)
jint GetVersion()
{ return functions->GetVersion(this); }
jclass DefineClass(const char *name, jobject loader, const jbyte* buf,
jsize bufLen)
{ return functions->DefineClass(this, name, loader, buf, bufLen); }
jclass FindClass(const char* name)
{ return functions->FindClass(this, name); }
...........................................
#endif /*__cplusplus*/
};
C++变量中的_JNIEnv结构体内,先定义了一个JNINativeInterface*型指针functions,当引用FindClass函数时,C++直接用JNIEnv->FindClass(name)即可,这时函数的return只会调用functions->FindClass(this, name);其中this相当于C环境中的env,既env->FindClass(str)
三、总结
C++环境中:使用类似env->FindClass(str)
方式引用。
C环境中:使用类似 (*env)->FindClass(env, str);
方式引用。