JNI允许您从本机代码内调用 Java 类方法。
要做到这一点,通常必须使用 Invocation API 在本机代码内创建和初始化一个 JVM。
下列是您可能决定从 C/C++ 代码调用Java 代码的典型情况:
1.希望实现的这部分代码是平台无关的,它将用于跨多种平台使用的功能。
2.需要在本机应用程序中访问用 Java 语言编写的代码或代码库。
3.希望从本机代码利用标准 Java 类库。
从C/C++ 程序调用 Java 代码的四个步骤:
1.编写 Java 代码。
这个步骤包含编写一个或多个 Java 类,这些类实现(或调用其它方法实现)您想要访问的功能。
2.编译 Java 代码。
在能够使用这些 Java 类之前,必须成功地将它们编译成字节码。
3.编写 C/C++ 代码。
这个代码将创建和实例化 JVM,并调用正确的 Java 方法。
4.运行本机 C/C++ 应用程序。
将运行应用程序以查看它是否正常工作。我们还将讨论一些用于处理常见错误的技巧。
步骤 1:编写Java 代码
我们从编写一个或多个 Java 源代码文件开始,这些文件将实现我们想要本机 C/C++ 代码使用的功能。
下面显示了一个 Java 代码示例JNI_cCalljava_test.java:
- package test;
-
- public class JNI_cCalljava_test {
-
- public static int intMethod(int n) {
- return n*n;
- }
-
- public static boolean booleanMethod(boolean bool) {
- return !bool;
- }
-
- }
注:JNI_cCalljava_test.java 实现了两个 static Java 方法:intMethod(intn) 和 booleanMethod(boolean bool)(分别在第 3 行和第 7 行)。static方法是一种不需要与对象实例关联的类方法。调用 static方法要更容易些,因为不必实例化对象来调用它们。
步骤 2:编译Java 代码
接下来,我们将 Java 代码编译成字节码。
完成这一步的方法之一是使用随SDK 一起提供的Java 编译器 javac。使用的命令是:
javac JNI_cCalljava_test.java
或者直接在eclipose中编写保存即可
步骤 3:编写 C/C++ 代码
即使是在本机应用程序中运行,所有 Java 字节码也必须在 JVM 中执行。
因此 C/C++ 应用程序必须包含用来创建和初始化 JVM 的调用。
为了方便我们,SDK 包含了作为共享库文件(jvm.dll 或 jvm.so)的 JVM,这个库文件可以嵌入到本机应用程序中。
让我们先从浏览一下 C 和 C++ 应用程序的整个代码开始,然后对两者进行比较。
带有嵌入式 JVM的 C 应用程序:
- #include <jni.h>
-
- #ifdef _WIN32
- #define PATH_SEPARATOR ';'
- #else
- #define PATH_SEPARATOR ':'
- #endif
-
-
-
- int main()
- {
-
-
-
-
-
-
-
- JavaVMOption options[1];
- JNIEnv *env;
- JavaVM *jvm;
- JavaVMInitArgs vm_args;
-
-
-
-
- long status;
- jclass cls;
- jmethodID mid;
- jint square;
- jboolean not;
-
-
-
-
-
-
-
-
-
- options[0].optionString = "-Djava.class.path=.";
- memset(&vm_args, 0, sizeof(vm_args));
- vm_args.version = JNI_VERSION_1_2;
- vm_args.nOptions = 1;
- vm_args.options = options;
-
-
-
-
- status = JNI_CreateJavaVM(&jvm, (void**)&env, &vm_args);
-
- if (status != JNI_ERR)
- {
-
-
-
-
-
-
-
- cls = (*env)->FindClass(env, "test/JNI_cCalljava_test");
- printf("test1,cls=%d...\n",cls);
-
- if(cls !=0)
- {
-
-
-
-
-
-
- mid = (*env)->GetStaticMethodID(env, cls, "intMethod", "(I)I");
-
-
-
-
-
- if(mid !=0)
- {
-
-
- square = (*env)->CallStaticIntMethod(env, cls, mid, 5);
- printf("Result of intMethod: %d\n", square);
- }
-
- mid = (*env)->GetStaticMethodID(env, cls, "booleanMethod", "(Z)Z");
- if(mid !=0)
- {
- not = (*env)->CallStaticBooleanMethod(env, cls, mid, 1);
- printf("Result of booleanMethod: %d\n", not);
- }
- }
-
- (*jvm)->DestroyJavaVM(jvm);
- return 0;
- }
- else
- return -1;
- }
带有嵌入式 JVM的 C++ 应用程序
- #include <jni.h>
-
- #ifdef _WIN32
- #define PATH_SEPARATOR ';'
- #else
- #define PATH_SEPARATOR ':'
- #endif
-
- int main()
- {
- JavaVMOption options[1];
- JNIEnv *env;
- JavaVM *jvm;
- JavaVMInitArgs vm_args;
- long status;
- jclass cls;
- jmethodID mid;
- jint square;
- jboolean not;
-
- options[0].optionString = "-Djava.class.path=.";
- memset(&vm_args, 0, sizeof(vm_args));
- vm_args.version = JNI_VERSION_1_2;
- vm_args.nOptions = 1;
- vm_args.options = options;
- status = JNI_CreateJavaVM(&jvm, (void**)&env, &vm_args);
-
- if (status != JNI_ERR)
- {
- cls = env->FindClass("Sample2");
- if(cls !=0)
- {
- mid = env->GetStaticMethodID(cls, "intMethod", "(I)I");
- if(mid !=0)
- {
- square = env->CallStaticIntMethod(cls, mid, 5);
- printf("Result of intMethod: %d\n", square);
- }
-
- mid = env->GetStaticMethodID(cls, "booleanMethod", "(Z)Z")
- if(mid !=0)
- {
- not = env->CallStaticBooleanMethod(cls, mid, 1);
- printf("Result of booleanMethod: %d\n", not);
- }
- }
-
- jvm->DestroyJavaVM();
- return 0;
- }
- else
- return -1;
- }
C 和 C++ 实现的比较
C 和C++ 代码几乎相同;唯一的差异在于用来访问 JNI 函数的方法。
在 C 中,为了取出函数指针所引用的值,JNI 函数调用前要加一个(*env)-> 前缀。
在 C++ 中,JNIEnv类拥有处理函数指针查找的内联成员函数。
因此,虽然这两行代码访问同一函数,但每种语言都有各自的语法,如下所示。
C 语法:
cls = (*env)->FindClass(env, "Sample2");
C++ 语法:
cls = env->FindClass("Sample2");
C 语法:
mid = (*env)->GetStaticMethodID(env, cls, "intMethod", "(I)I");
C++ 语法:
mid = env->GetStaticMethodID(cls, "intMethod", "(I)I");
C 语法:
square = env->CallStaticIntMethod(cls, mid, 5);
C++ 语法:
square = (*env)->CallStaticIntMethod(env, cls, mid, 5);
C 语法:
(*jvm)->DestroyJavaVM(jvm);
C++ 语法:
jvm->DestroyJavaVM();
编译方式
gcc cCall.c -o cCall -I/usr/lib/jvm/java-1.6.0/include/ -I/usr/lib/jvm/java-1.6.0/include/linux -I/usr/lib/jvm/java-1.6.0/jre/lib/amd64/server -L/usr/lib/jvm/java-1.6.0/jre/lib/amd64/server -ljvm
c语言使用gcc c++使用g++