JNI
JNI是Java Native Interface的缩写,它提供了若干的API实现了Java和其他语言的通信(主要是C&C++)
什么时候使用JNI
当Java层无法满足功能需求的时候,或者对性能要求很高时候,往往会使用JNI
NDK
工具的集合。帮助开放者快速开放C/C++ 动态库的工具。
JNI的开发流程
步骤:
- 编写native 方法
- 在cpp目录下实现native方法,对应的方法名为(java_类的全名_方法名)
- 生成一个动态库
- 在java中加载动态库
- 触发native函数
JNIEnv
JNIEnv实际上表示的是Java运行环境,在JNI中经常使用的env
其实是一个指针,通过它就可以对Java端进行操作。例如,创建Java对象,调用Java方法等。
- 在C中,env 是二级指针
- 在C++中,env 是一级指针
每个jni中的native方法,都至少有两个参数(JNIEnv * , jclass/jobject)
jclass: native 静态方法
jobject: native 非静态方法
Java和JNI中类型对应关系
Java类型 | 本地类型 | 字节(bit) |
---|---|---|
boolean | jboolean | 8, unsigned |
byte | jbyte | 8 |
char | jchar | 16, unsigned |
short | jshort | 16 |
int | jint | 32 |
long | jlong | 64 |
float | jfloat | 32 |
double | jdouble | 64 |
void | void | n/a |
JNI中C调用Java
调用普通方法
步骤
- 找到java对应的class,传入的参数是JNIEnv指针和Java文件的完整路径
jclass jclazz = (*env)->FindClass(env,"com/demo/tianyl/CCallJava");
- 找到对应的方法,传入的参数是JNIEnv指针和Java的class对象还有方法签名,其中方法签名按照
(参数)返回值
的字符串传入
jmethodID methodid = (*env)->GetMethodID(env, jclazz,"callJava","()V");
Java中方法签名对应关系
特殊字符 | 数据类型 | 说明 |
---|---|---|
V | void | 一般用于方法的返回值 |
Z | boolean | |
B | byte | |
C | char | |
S | short | |
I | int | |
J | long | |
F | float | |
D | double | |
[ | 数组 | 以[开头,配合其他的特殊字符,表示对应数据类型的数组,几个[表示几维数组 |
L全类名; | 引用类型 | 以L开头,;结尾,中间是引用类型的全类名 |
- 调用方法
(*env)->CallVoidMethod(env,obj,methodid);
在调用方法时,因为这里是非静态的方法,所以传入对应的Java object对象,并且使用CallVoidMethod
,如果是静态的方法,则是传入class对象,并使用CallStaticObjectMethod
调用构造方法
调用构造方法其实就是调用了一个静态的普通方法,大体流程基本一致
步骤
- 找到java对应的class,传入的参数是JNIEnv指针和Java文件的完整路径
jclass jclazz = (*env)->FindClass(env,"com/demo/tianyl/Test");
- 找到对应的方法,这里是静态方法,所以需要使用
GetStaticMethodID
,传入的参数是JNIEnv指针和Java的class对象还有方法签名,其中方法签名按照(参数)返回值
的字符串传入
jmethodID methodid = (*env)->GetStaticMethodID(env, jclazz,"<init>","()V");
- 调用方法
如果是需要使用构造方法创建对象,那么就不能直接使用CallVoidMethod
之类的方法了,需要使用NewObject
方法
jobject jobj = (*env)->NewObject(env,obj,methodid);