步骤 2:编译 Java 代码
接下来,我们需要将 Java 代码编译成字节码。
完成这一步的方法之一是使用随SDK一起提供的Java编译器javac。
用来将 Java 代码编译成字节码的命令是:
cd test
javac JNI_javaCallc_test.java
如果是在eclipse环境下编写的以上代码,文件保存时会自动在工程目录的bin下生成以上java文件
步骤 3:创建 C/C++ 头文件
第三步是创建 C/C++ 头文件,它定义本机函数说明。
完成这一步的方法之一是使用 javah.exe,它是随 SDK 一起提供的本机方法 C 存根生成器工具。
这个工具被设计成用来创建头文件,该头文件为在 Java 源代码文件中所找到的每个 native 方法定义 C 风格的函数。
这里使用的命令是:
cd test
javah -classpath . test.JNI_javaCallc_test
注意.和test之间有空格
会生成以下文件:
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class test_JNI_javaCallc_test */
#ifndef _Included_test_JNI_javaCallc_test
#define _Included_test_JNI_javaCallc_test
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: test_JNI_javaCallc_test
* Method: intMethod
* Signature: (I)I
*/
JNIEXPORT jint JNICALL Java_test_JNI_1javaCallc_1test_intMethod
(JNIEnv *, jobject, jint);
/*
* Class: test_JNI_javaCallc_test
* Method: booleanMethod
* Signature: (Z)Z
*/
JNIEXPORT jboolean JNICALL Java_test_JNI_1javaCallc_1test_booleanMethod
(JNIEnv *, jobject, jboolean);
/*
* Class: test_JNI_javaCallc_test
* Method: stringMethod
* Signature: (Ljava/lang/String;)Ljava/lang/String;
*/
JNIEXPORT jstring JNICALL Java_test_JNI_1javaCallc_1test_stringMethod
(JNIEnv *, jobject, jstring);
/*
* Class: test_JNI_javaCallc_test
* Method: intArrayMethod
* Signature: ([I)I
*/
JNIEXPORT jint JNICALL Java_test_JNI_1javaCallc_1test_intArrayMethod
(JNIEnv *, jobject, jintArray);
#ifdef __cplusplus
}
#endif
#endif
如您可能已经注意到的那样,JNI_javaCallc_test.h 中的 C/C++ 函数说明和 JNI_javaCallc_test.java 中的 Java native 方法声明有很大差异。
1、JNIEXPORT 和 JNICALL 是用于导出函数的、依赖于编译器的指示符。
2、返回类型、参数类型是映射到 Java 类型的 C/C++ 类型,比如:jstring,jint
现在来介绍下JNI里的数据类型:
在C++里,编译器会很据所处的平台来为一些基本的数据类型来分配长度,因此也就造成了平台不一致性,而这个问题在Java中则不存在,因为有JVM的缘故,所以Java中的基本数据类型在所有平台下得到的都是相同的长度,比如int的宽度永远都是32位。基于这方面的原因,java和c++的基本数据类型就需要实现一些mapping,保持一致性。
下面的表可以概括:下标列举了常见的c/c++到到java的类型映射表。
Java类型 | 本地类型 | JNI中定义的别名 |
int | long | jint |
long | _int64 | jlong |
byte | signed char | jbyte |
boolean | unsigned char | jboolean |
char | unsigned short | jchar |
short | short | jshort |
float | float | jfloat |
double | double | jdouble |
Object | _jobject* | jobject |
除了 Java 声明中的一般参数以外,所有这些函数的参数表中都有一个指向 JNIEnv 和 jobject 的指针。
指向 JNIEnv 的指针实际上是一个指向函数指针表的指针。
正如将要在步骤4 中看到的,这些函数提供各种用来在C和C++中操作Java数据的能力。
jobject 参数引用当前对象
因此,如果C或C++代码需要引用Java函数,则这个jobject 充当引用或指针,返回调用的 Java 对象。
函数名本身是由前缀“Java_”加全限定类名,再加下划线和方法名构成的。