java/android下JNI编程教程


java/android下JNI编程教程 课程大纲: 1、java jni简介 2、windows下java JNI编程技巧——JAVA调用c/c++(0) 3、windows下java JNI编程技巧——JAVA调用c/c++(1) 4、windows下java JNI编程技巧——JAVA调用c/c++(2) 5、windows下java JNI编程技巧——JAVA调用c/c++(3) 6、windows下java JNI编程技巧——JAVA调用c/c++(4) 7、从 C/C++ 程序调用 Java 代码 【教程一】JAVA JNI简介: Java 本机接口(Java Native Interface (JNI))是一个本机编程接口, 它是 Java 软件开发工具箱(Java SoftwareDevelopment Kit (SDK))的一部 分,JNI它提供了若干的API,实现了和Java和其他语言的通信(主要是C&C++)。 JNI允许Java代码使用以其它语言(譬如 C 和 C++)编写的代码和代码库。 Invocation API(JNI的一部分)可以用来将Java虚拟机(JVM)嵌入到本 机应用程序中,从而允许程序员从本机代码内部调用Java 代码。 也许不少人觉得Java已经足够强大,为什么要需要JNI这种东西呢? 我们知道Java是一种平台无关性的语言,平台对于上层的java代码来说是 透明的,所以在多数时间我们是不需要JNI的,但是假如你遇到了如下的三种情 况之一呢? 1. 你的Java代码,需要得到一个文件的属性。但是你找遍了JDK帮助文档也 找不到相关的API。 2. 在本地还有一个别的系统,不过他不是Java语言实现的,这个时候你的 老板要求你把两套系统整合到一起。 3. 你的Java代码中需要用到某种算法,不过算法是用C实现并封装在动态链 接库文件(DLL)当中的。 对于上述的三种情况,如果没有JNI的话,那就会变得异常棘手了。就算找到解 决方案了,也是费时费力。其实说到底还是会增加开发和维护的成本。 二、环境需求 JNI 最常见的两个应用:从Java程序调用C/C++,以及从C/C++程序调用Java代 码 1、需要下列工具与组件: javac.exe: Java 编译器:随 SDK(Java 2 SDK及以上) 一起提供的 。 java.exe: Java 虚拟机(JVM):随 SDK 一起提供的 。 javah.exe: 本机方法 C 文件生成器:随 SDK 一起提供的 。 2、定义JNI的库文件和本机头文件: jni.h (C 头文件)、jvm.lib 和 jvm.dll (window下)或 libjvm.so 文 件(linux下),这些文件都是随 SDK 一起提供的。 3、能够创建共享库的 C 和 C++ 编译器。 最常见的两个 C 编译器是用于Windows的Visual C++和用于基于UNIX系统的 gcc/cc。 因此,后面我们将会介绍在两种环境下的JNI编程例子。 【教程二】windows下java JNI编程技巧——JAVA调用c/c++(0) 一、使用情况 当无法用 Java 语言编写整个应用程序时,JNI 允许您使用本机代 码。 在下列典型情况下,您可能决定使用本机代码: 1、希望用更低级、更快的编程语言去实现对时间有严格要求的代码。 2、希望从 Java 程序访问旧代码或代码库。 3、需要标准 Java 类库中不支持的依赖于平台的特性。 二、所需软件 eclipse3.4.1、JDK6、VC6.0 三、步骤分析 从 Java 程序调用 C 或 C ++ 代码的过程由六个步骤组成: 我们将在下面几页中深入讨论每个步骤,但还是先让我们迅速地浏览一下它们: 1、编写 Java 代码。 我们将从编写 Java 类开始,这些类执行三个任务: 1)声明将要调用的本机方法; 2)装入包含本机代码的共享库; 3)然后调用该本机方法。 2、编译 Java 代码。 在使用 Java 类之前,必须成功地将它们编译成字节码。 3、创建 C/C++头文件。 C/C++头文件将声明想要调用的本机函数说明。 然后,这个头文件与 C/C++ 函数实现(请参阅步骤 4)一起来创建共享库 (请参阅步骤 5)。 4、编写 C/C++ 代码。 这一步实现 C 或 C++ 源代码文件中的函数。 C/C++ 源文件必须包含步骤 3 中创建的头文件。 5、创建共享库文件。 从步骤 4 中创建的 C 源代码文件来创建共享库文件。 6、运行 Java 程序。 运行该代码,并查看它是否有用。我们还将讨论一些用于解决常见错误的技 巧。 【教程三】windows下java JNI编程技巧——JAVA调用c/c++(1) 步骤 1:编写 Java 代码 我们从编写 Java 源代码文件开始,它将声明本机方法(或方法),装入包含 本机代码的共享库,然后实际调用本机方法。 这里是名为 JNI_javaCallc_test: 直接使用文本编辑器或在 ecilpos 中建立工程敲入以下代码: [cpp] view plaincopy 1. package test; 2. 3. public class JNI_javaCallc_test { 4. 5. //c/c++本地方法 6. public native int intMethod(int n); 7. 8. public native boolean booleanMethod(boolean bool); 9. 10. public native String stringMethod(String text); 11. 12. public native int intArrayMethod(int[] intArray); 13. 14. //java main 方法 15. public static void main(String[] args){ 16. //包含 c 语言动态库 17. System.loadLibrary("test_JNI_javaCallc_test"); 18. 19. JNI_javaCallc_test sample = new JNI_javaCallc_test(); 20. 21. int square = sample.intMethod(5); 22. 23. boolean bool = sample.booleanMethod(true); 24. 25. String text =sample.stringMethod("JAVA"); 26. 27. int sum = sample.intArrayMethod(new int[] { 1, 1, 2, 3, 5, 8, 13 }); 28. 29. System.out.println("intMethod: " + square); 30. 31. System.out.println("booleanMethod: " + bool); 32. 33. System.out.println("stringMethod: " + text); 34. 35. System.out.println("intArrayMethod: " + sum); 36. } 37. } 这段代码做了些什么? 首先,请注意对 native 关键字的使用,它只能随方法一起使用。 native 关键字告诉 Java 编译器:方法是用 Java 类之外的本机代码实现 的,但其声明却在 Java 中。只能在 Java 类中声明本机方法,而不能实现它 (但是不能声明为抽象的方法,使用 native 关键字即可),所以本机方法不能 拥有方法主体。 现在,让我们逐行研究一下代码: 从第 6 行到第 12 行,我们声明了四个 native 方法。 在第 17 行,我们装入了包含这些本机方法的实现的共享库文件。(到步骤 5 时,我们将创建该共享库文件。) 最终,从第 21 行到第 27 行,我们调用了本机方法。 注:这个操作和调用非本机 Java 方法的操作没有差异。 【教程四】windows下java JNI编程技巧——JAVA调用c/c++(2) 步骤 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 之间有空格 会生成以下文件: [cpp] view plaincopy 1. /* DO NOT EDIT THIS F ILE - it is machine generated */ 2. #include 3. /* Header for class test_JNI_javaCallc_test */ 4. 5. #ifndef _Included_test_JNI_javaCallc_test 6. #define _Included_test_JNI_javaCallc_test 7. #ifdef __cplusplus 8. extern "C" { 9. #endif 10. /* 11. * Class: test_JNI_javaCallc_test 12. * Method: intMethod 13. * Signature: (I)I 14. */ 15. JNIEXPORT jint JNICALL Java_test_JNI_1javaCallc_1test_intMethod 16. (JNIEnv *, jobject, jint); 17. 18. /* 19. * Class: test_JNI_javaCallc_test 20. * Method: booleanMethod 21. * Signature: (Z)Z 22. */ 23. JNIEXPORT jboolean JNICALL Java_test_JNI_1javaCallc_1test_booleanMetho d 24. (JNIEnv *, jobject, jboolean); 25. 26. /* 27. * Class: test_JNI_javaCallc_test 28. * Method: stringMethod 29. * Signature: (Ljava/lang/String;)Ljava/lang/String; 30. */ 31. JNIEXPORT jstring JNICALL Java_test_JNI_1javaCallc_1test_stringMethod 32. (JNIEnv *, jobject, jstring); 33. 34. /* 35. * Class: test_JNI_javaCallc_test 36. * Method: intArrayMethod 37. * Signature: ([I)I 38. */ 39. JNIEXPORT jint JNICALL Java_test_JNI_1javaCallc_1test_intArrayMethod 40. (JNIEnv *, jobject, jintArray); 41. 42. #ifdef __cplusplus 43. } 44. #endif 45. #endif 46. 关于 C/C++ 头文件 如您可能已经注意到的那样,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 JNI 的设计者其实已经帮我们取好了相应的别名以方便记忆。如果想了解一些更 加细致的信息,可以去看一些 jni.h 这个头文件,各种数据类型的定义以及别 名就被定义在这个文件中。 除了 Java 声明中的一般参数以外,所有这些函数的参数表中都有一个指向 JNIEnv 和 jobject 的指针。 指向 JNIEnv 的指针实际上是一个指向函数指针表的指针。 正如将要在步骤 4 中看到的,这些函数提供各种用来在 C 和 C++中操作 Java 数 据的能力。 jobject 参数引用当前对象 因此,如果 C 或 C++代码需要引用 Java 函数,则这个jobject 充当引用或指针, 返回调用的 Java 对象。 函数名本身是由前缀“Java_”加全限定类名,再加下划线和方法名构成的。 【教程五】windows下java JNI编程技巧——JAVA调用c/c++(3) 步骤 4:编写 C/C++ 代码 当谈到编写 C/C++ 函数实现时,有一点需要牢记:说明必须和 JNI_javaCallc_test.h 的函数声明完全一样。 我们将研究用于 C 实现和 C++ 实现的完整代码,然后讨论两者之间的差 异。 C 函数实现 以下是 JNI_javaCallc_test.c,它是用 C 编写的实现: [cpp] view plaincopy 1. #include 2. /* Header for class test_JNI_javaCallc_test */ 3. 4. 5. /* 6. * Class: test_JNI_javaCallc_test 7. * Method: intMethod 8. * Signature: (I)I 9. */ 10. JNIEXPORT jint JNICALL Java_test_JNI_1javaCallc_1test_intMethod(JNIEnv *env, jobject obj, jint num) 11. { 12. return num * num; 13. } 14. 15. /* 16. * Class: test_JNI_javaCallc_test 17. * Method: booleanMethod 18. * Signature: (Z)Z 19. */ 20. JNIEXPORT jboolean JNICALL Java_test_JNI_1javaCallc_1test_booleanMetho d 21. (JNIEnv *env, jobject obj, jboolean boolean) { 22. return!boolean; 23. } 24. /* 25. * Class: test_JNI_javaCallc_test 26. * Method: stringMethod 27. * Signature: (Ljava/lang/String;)Ljava/lang/String; 28. */ 29. JNIEXPORT jstring JNICALL Java_test_JNI_1javaCallc_1test_stringMethod 30. (JNIEnv *env, jobject obj, jstring string) 31. { 32. const char *str = (*env)->GetStringUTFChars(env, string, 0); 33. char cap[128]; 34. strcpy(cap, str); 35. (*env)->ReleaseStringUTFChars(env, string, str); 36. return (*env)->NewStringUTF(env, strupr(cap)); 37. } 38. 39. /* 40. * Class: test_JNI_javaCallc_test 41. * Method: intArrayMethod 42. * Signature: ([I)I 43. */ 44. JNIEXPORT jint JNICALL Java_test_JNI_1javaCallc_1test_intArrayMethod 45. (JNIEnv *env, jobject obj, jintArray array) 46. { 47. int i, sum = 0; 48. jsize len = (*env)->GetArrayLength(env, array); 49. jint*body = (*env)->GetIntArrayElements(env, array, 0); 50. for(i=0; iReleaseIntArrayElements(env, array, body, 0); 54. return sum; 55. } C++ 函数实现 [cpp] view plaincopy 1. #include 2. /* Header for class test_JNI_javaCallc_test */ 3. 4. 5. /* 6. * Class: test_JNI_javaCallc_test 7. * Method: intMethod 8. * Signature: (I)I 9. */ 10. JNIEXPORT jint JNICALL Java_test_JNI_1javaCallc_1test_intMethod(JNIEnv *env, jobject obj, jint num) 11. { 12. return num * num; 13. } 14. 15. /* 16. * Class: test_JNI_javaCallc_test 17. * Method: booleanMethod 18. * Signature: (Z)Z 19. */ 20. JNIEXPORT jboolean JNICALL Java_test_JNI_1javaCallc_1test_booleanMetho d 21. (JNIEnv *env, jobject obj, jboolean boolean) { 22. return!boolean; 23. } 24. /* 25. * Class: test_JNI_javaCallc_test 26. * Method: stringMethod 27. * Signature: (Ljava/lang/String;)Ljava/lang/String; 28. */ 29. JNIEXPORT jstring JNICALL Java_test_JNI_1javaCallc_1test_stringMethod 30. (JNIEnv *env, jobject obj, jstring string) 31. { 32. constchar *str = env->GetStringUTFChars(string, 0); 33. char cap[128]; 34. strcpy(cap, str); 35. env->ReleaseStringUTFChars(string, str); 36. returnenv->NewStringUTF(strupr(cap)); 37. } 38. 39. /* 40. * Class: test_JNI_javaCallc_test 41. * Method: intArrayMethod 42. * Signature: ([I)I 43. */ 44. JNIEXPORT jint JNICALL Java_test_JNI_1javaCallc_1test_intArrayMethod 45. (JNIEnv *env, jobject obj, jintArray array) 46. { 47. int i,sum = 0; 48. jsizelen = env->GetArrayLength(array); 49. jint*body = env->GetIntArrayElements(array, 0); 50. for(i=0; iReleaseIntArrayElements(array, body, 0); 55. returnsum; 56. } C 和 C++ 函数实现的比较 唯一的差异在于用来访问 JNI 函数的方法。 在 C 中,JNI 函数调用由“(*env)->”作前缀,目的是为了取出函数指针所引 用的值。 在 C++ 中,JNIEnv 类拥有处理函数指针查找的内联成员函数。 下面将说明这个细微的差异,其中,这两行代码访问同一函数,但每种语言都 有各自的语法 C 语法: jsize len = (*env)->GetArrayLength(env,array); C++语法: jsize len =env->GetArrayLength(array); 【教程六】windows下java JNI编程技巧——JAVA调用c/c++(4) 步骤 5:创建共享库文件 接下来,我们创建包含本机代码的共享库文件。 大多数 C 和 C++ 编译器除了可以创建机器代码可执行文件以外,也可以创建 共享库文件。 用来创建共享库文件的命令取决于您使用的编译器。 下面是在 Windows执行的命令。 Windows: 使用visual studio commandprompt工具cl.exe cl -I"C:\Program Files\Java\jdk1.6.0_10\include" -I"C:\Program Files\Java\jdk1.6.0_10\include\win32" -LD test_JNI_javaCallc_test.c -Fe test_JNI_javaCallc_test.dll 也可以使用vc6.0 直接建立动态库 编译的时候需要jni相关的头文件和库文件,在vc6.0 的的搜索路径加入与java 有关的两个路径即可即可 Tools->sptions->Directories Linux:使用gcc工具 gcc -c -fPIC -I/usr/java/jdk1.6.0_22/include/ -I/usr/java/jdk1.6.0_22/include/linux/ Sample1.c gcc -shared -fPIC -o libSample1.so Sample1.o 步骤 6:运行 Java 程序 最后一步是运行 Java 程序,并确保代码正确工作。 因为必须在 Java 虚拟机中执行所有 Java 代码,所以需要使用 Java 运行时 环境。 完成这一步的方法之一是使用 java,它是随 SDK 一起提供的 Java 解释器。 所使用的命令是: java -cp . test.test_JNI_javaCallc_test 或者直接在eclipose中运行即可 输出: intMethod: 25 booleanMethod: false stringMethod: JAVA intArrayMethod: 33 【教程七】从 C/ C++ 程序调用 Java 代码 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: [java] view plaincopy 1. package test; 2. 3. public class JNI_cCalljava_test { 4. 5. public static int intMethod(int n) { 6. return n*n; 7. } 8. 9. public static boolean booleanMethod(boolean bool) { 10. return !bool; 11. } 12. 13. } 注:JNI_cCalljava_test.java 实现了两个 static Java 方法:intMethod(intn) 和 booleanMethod(boolean bool)(分别在第 3 行和第 7 行)。static 方法 是一种不需要与对象实例关联的类方法。调用 static 方法要更容易些,因为不 必实例化对象来调用它们。 步骤 2:编译 Java 代码 接下来,我们将 Java 代码编译成字节码。 完成这一步的方法之一是使用随 SDK 一起提供的 Java 编译器 javac。使用的 命令是: JNI_cCalljava_test.java 或者直接在 eclipose 中编写保存即可 步骤 3:编写 C/C++ 代码 即使是在本机应用程序中运行,所有 Java 字节码也必须在 JVM 中执行。 因此 C/C++ 应用程序必须包含用来创建和初始化 JVM 的调用。 为了方便我们,SDK 包含了作为共享库文件(jvm.dll 或 jvm.so)的 JVM,这 个库文件可以嵌入到本机应用程序中。 让我们先从浏览一下 C 和 C++ 应用程序的整个代码开始,然后对两者进行比 较。 带有嵌入式 JVM 的 C 应用程序: [cpp] view plaincopy 1. #include 2. //jni.h 文件包含在 C 代码中所需要的 JNI 的所有类型和函数定义 3. #ifdef _WIN32 4. #define PATH_SEPARATOR ';' 5. #else 6. #define PATH_SEPARATOR ':' 7. #endif 8. //1.包括准备本机应用程序以处理 Java 代码 9. //2.将 JVM 嵌入本机应用程序 10. //3.然后从该应用程序内找到并调用 Java 方法。 11. int main() 12. { 13. /* 14. 接下来,声明所有希望在程序中使用的变量。 15. JavaVMOption options[] 具有用于 JVM 的各种选项设置。 16. 当声明变量时,确保所声明的 JavaVMOption options[] 数组足够大,以便能容纳您希望使 用的所有选项。 17. 在本例中,我们使用的唯一选项就是类路径选项。 18. 因为在本示例中,我们所有的文件都在同一目录中,所以将类路径设置成当前目录。 19. 可以设置类路径,使它指向任何您希望使用的目录结构。*/ 20. JavaVMOption options[1]; 21. JNIEnv *env; 22. JavaVM *jvm; 23. JavaVMInitArgs vm_args; 24. /*JNIEnv *env 表示 JNI 执行环境。 25. JavaVM jvm 是指向 JVM 的指针,我们主要使用这个指针来创建、初始化和销 毁 JVM。 26. JavaVMInitArgs vm_args 表示可以用来初始化 JVM 的各种 JVM 参数。*/ 27. 28. long status; 29. jclass cls; 30. jmethodID mid; 31. jint square; 32. jboolean not; 33. 34. /*avaVMInitArgs 结构表示用于 JVM 的初始化参数。 35. 在执行 Java 代码之前,可以使用这些参数来定制运行时环境。 36. 正如您所见,这些选项是一个参数,而 Java 版本是另一个参数。 37. 按如下所示设置了这些参数:*/ 38. 39. /*为 JVM 设置类路径,以使它能找到所需要的 Java 类。 40. 在这个特定示例中,因为 Sample2.class 和 Sample2.exe 都位于同一目录中,所以将类路 径设置成当前目录。 41. 我们用来为 Sample2.c 设置类路径的代码如下所示:*/ 42. options[0].optionString = "-Djava.class.path=."; 43. memset(&vm_args, 0, sizeof(vm_args)); 44. vm_args.version = JNI_VERSION_1_2; 45. vm_args.nOptions = 1; 46. vm_args.options = options; 47. 48. /*创建 JVM 49. 处理完所有设置之后,现在就准备创建 JVM 了。先从调用方法开始 50. 如果成功,则这个方法返回零,否则,如果无法创建 JVM,则返回 JNI_ERR。*/ 51. status = JNI_CreateJavaVM(&jvm, (void**)&env, &vm_args); 52. 53. if (status != JNI_ERR) 54. { 55. /* 56. 查找并装入 Java 类 57. 一旦创建了 JVM 之后,就可以准备开始在本机应用程序中运行 Java 代码。 58. 首先,需要使用 FindClass() 函数查找并装入 Java 类,如下所示: 59. cls 变量存储执行 FindClass() 函数后的结果,如果找到该类,则 cls 变量表示该 Java 类 的句柄, 60. 如果不能找到该类,则 cls 将为零。 61. */ 62. cls = (*env)->FindClass(env, "test/JNI_cCalljava_test"); 63. printf("test1,cls=%d...\n",cls); 64. 65. if(cls !=0) 66. { 67. /* 68. 查找 Java 方法 69. 接下来,我们希望用 GetStaticMethodID() 函数在该类中查找某个方法。 70. 我们希望查找方法 intMethod,它接收一个 int 参数并返回一个 int。 71. 以下是查找 intMethod 的代码: 72. */ 73. mid = (*env)->GetStaticMethodID(env, cls, "intMethod", "(I )I"); 74. /* 75. mid 变量存储执行 GetStaticMethodID() 函数后的结果。 76. 如果找到了该方法,则 mid 变量表示该方法的句柄。 77. 如果不能找到该方法,则 mid 将为零。 78. */ 79. if(mid !=0) 80. { 81. /*CallStaticIntMethod() 方法接受 cls(表示类)、mid(表示方法)以及用于该方法一 个或多个参数。 82. 在本例中参数是 int 5。*/ 83. square = (*env)->CallStaticIntMethod(env, cls, mid, 5) ; 84. printf("Result of intMethod: %d\n", square); 85. } 86. 87. mid = (*env)->GetStaticMethodID(env, cls, "booleanMethod", "(Z)Z"); 88. if(mid !=0) 89. { 90. not = (*env)->CallStaticBooleanMethod(env, cls, mid, 1 ); 91. printf("Result of booleanMethod: %d\n", not); 92. } 93. } 94. 95. (*jvm)->DestroyJavaVM(jvm); 96. return 0; 97. } 98. else 99. return -1; 100. } 带有嵌入式 JVM 的 C++ 应用程序 [cpp] view plaincopy 1. #include 2. 3. #ifdef _WIN32 4. #define PATH_SEPARATOR ';' 5. #else 6. #define PATH_SEPARATOR ':' 7. #endif 8. 9. int main() 10. { 11. JavaVMOption options[1]; 12. JNIEnv *env; 13. JavaVM *jvm; 14. JavaVMInitArgs vm_args; 15. long status; 16. jclass cls; 17. jmethodID mid; 18. jint square; 19. jboolean not; 20. 21. options[0].optionString = "-Djava.class.path=."; 22. memset(&vm_args, 0, sizeof(vm_args)); 23. vm_args.version = JNI_VERSION_1_2; 24. vm_args.nOptions = 1; 25. vm_args.options = options; 26. status = JNI_CreateJavaVM(&jvm, (void**)&env, &vm_args); 27. 28. if (status != JNI_ERR) 29. { 30. cls = env->FindClass("Sample2"); 31. if(cls !=0) 32. { 33. mid = env->GetStaticMethodID(cls, "intMethod", "(I)I"); 34. if(mid !=0) 35. { 36. square = env->CallStaticIntMethod(cls, mid, 5); 37. printf("Result of intMethod: %d\n", square); 38. } 39. 40. mid = env->GetStaticMethodID(cls, "booleanMethod", "(Z)Z") 41. if(mid !=0) 42. { 43. not = env->CallStaticBooleanMethod(cls, mid, 1); 44. printf("Result of booleanMethod: %d\n", not); 45. } 46. } 47. 48. jvm->DestroyJavaVM(); 49. return 0; 50. } 51. else 52. return -1; 53. } 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(); 步骤 4:运行应用程序 现在准备运行这个 C 应用程序,并确保代码正常工作。当运行 Sample2.exe 时,应该可以得到如下结果: windows: 使用 vc6.0 建一个普通的 C 语言工程 头文件路径设置同 Java 调用 C 语言里的设置 连接时需要 jvm.lib 支持 这里需要右击建立的工程,单击设置(Settings),link 选项栏将数据库路径添 加进来 C:"\Program Files"\Java\jdk1.6.0_10\lib\jvm.lib 在下面的 project options 中加入以上语句,用空格隔开,programe Files 用 双引号引起来 运行时需要 jvm.dll 动态库的支持,需要在系统环境变量中增加以下路径: C:\Program Files\Java\jdk1.6.0_10\jre\bin\server 方法:右击 我的电脑-》属性-》高级-》环境变量-》PATH 编辑,在原有环境 变量的基础上增加以上路径,注意用";"号隔开 将 eclipose 生成的 java 代码放在 JNI_cCalljava_test.exe 同目录下(注意按 照把报名文件夹也拷过去) E:\Sample2>JNI_cCalljava_test.exe Result of intMethod: 25 Result of booleanMethod: 0 以上教程由凌阳教育 Android 培训讲师-徐哥提供,凌阳教育专注于嵌入式 Linux 培训, Android 培训领域,拥有十多年的高校嵌入式师资培训经验,成为中国高校嵌入式培训第一 品牌! 欢迎访问凌阳教育官方网站:http://www.sunplusedu.com
还剩26页未读

继续阅读

下载pdf到电脑,查找使用更方便

pdf的实际排版效果,会与网站的显示效果略有不同!!

需要 8 金币 [ 分享pdf获得金币 ] 1 人已下载

下载pdf

pdf贡献者

goway5151

贡献于2013-10-11

下载需要 8 金币 [金币充值 ]
亲,您也可以通过 分享原创pdf 来获得金币奖励!
下载pdf