在Java代码中通过JNI调用C函数的步骤如下:
第一步:编写Java代码
第二步:编译Java代码(javac Java文件)
第三步:生成C代码头文件(javah java类名,自动生成)
第四步:编写C代码(实现C代码头文件里面的函数)
第五步:生成C共享库(使用工具编译生成C共享库,win下面为dll文件,Linux下面为so文件)
第六步:运行Java程序(java 类名)
一、编写Java代码
首先编写调用C语言的Java源代码HelloJNI.java
public class HelloJNI { native void printHello(); native void printString(String str); static { System.loadLibrary("hellojni"); } public static void main(String[] args) { // TODO Auto-generated method stub HelloJNI myJNI = new HelloJNI(); myJNI.printHello(); myJNI.printString("Hello world form printString function!"); } }
说明:
1. 在Java类中,使用”native”关键字,声明本地方法,该方法与用C/C++编写的JNI本地函数相对应。”native”关键字告知Java编译器,在Java代码中带有该关键字的方法只是声明,具体由C/C++等其它语言编写实现。
2. 在Java类中声明了本地方法之后,接下来,调用System.loadLibrary()方法,加载具体实现本地方法的C运行库(在Java中加载本地运行库通常使用静态块(static block))。System.loadLibrary()方法加载由字符串参数指定的本地库,在不同操作系统平台下,加载的C运行库不同。在Window下面,调用System.loadLibrary(“hellojni”),则hellojni.dll会被加载;在Linux下面,则会加载libhellojni.so文件。
二、编译Java代码
使用javac编译java源代码
javac HelloJNI.java
三、生成C代码头文件
使用javah自动生成C代码头文件
javah HelloJNI
生成的头文件HelloJNI.h见下:
/* DO NOT EDIT THIS FILE - it is machine generated */ #include <jni.h> /* Header for class HelloJNI */ #ifndef _Included_HelloJNI #define _Included_HelloJNI #ifdef __cplusplus extern "C" { #endif /* * Class: HelloJNI * Method: printHello * Signature: ()V */ JNIEXPORT void JNICALL Java_HelloJNI_printHello (JNIEnv *, jobject); /* * Class: HelloJNI * Method: printString * Signature: (Ljava/lang/String;)V */ JNIEXPORT void JNICALL Java_HelloJNI_printString (JNIEnv *, jobject, jstring); #ifdef __cplusplus } #endif #endif
四、编写C代码
实现C头文件中的方法,hellojni.c文件见下:
#include "HelloJNI.h" #include <stdio.h> JNIEXPORT void JNICALL Java_HelloJNI_printHello(JNIEnv *env, jobject obj) { printf("Hello world! "); return ; } JNIEXPORT void JNICALL Java_HelloJNI_printString(JNIEnv *env, jobject obj, jstring string) { const char * str = (*env)->GetStringUTFChars(env,string,0); printf("%s! ",str); return ; }
说明:
1、GetStringUTFChars ()是JNI函数,用来将Java字符串转换成C语言字符串。JNI提供了多种JNI函数,用来处理C字符串与Java字符串的转换,具体参见http://blog.csdn.net/qinjuning/article/details/7595104
五、生成C共享库
在Linux下测试,书写makefile文件,编译刚才所写的C代码,生成.so文件;
makefile文件见下:
libhellojni.so:hellojni.o makefile gcc -Wall -rdynamic -shared -o libhellojni.so hellojni.o hellojni.o:hellojni.c HelloJNI.h gcc -Wall -c hellojni.c -I./ -I/usr/lib/jdk/include -I/usr/lib/jdk/include/linux -fPIC cl: rm -rf *.o *.so
使用make命令编译
make
说明:
1、生成的so文件必须命令为libhellojni.so,与java代码中的System.loadLibrary("hellojni")对应;
2、在编译时,必须加上-fPIC,否则,编译时报以下错误:
/usr/bin/ld: hellojni.o: relocation R_X86_64_32 against `a local symbol' can not be used when making a shared object; recompile with -fPIC
hellojni.o: could not read symbols: Bad value
collect2: ld 返回 1
make: *** [hellojni.so] 错误 1
3、gcc和rm前面必须是一个tab空格,否则makefile格式不正确;
六、运行java程序
在运行java程序之前,要把生成的so文件加入LD_LIBRARY_PATH中,如下:
export LD_LIBRARY_PATH=./
否则,在运行java程序的时候报错,错误信息见下:
Exception in thread "main" java.lang.UnsatisfiedLinkError: no hellojni in java.library.path at java.lang.ClassLoader.loadLibrary(ClassLoader.java:1738) at java.lang.Runtime.loadLibrary0(Runtime.java:823) at java.lang.System.loadLibrary(System.java:1028) at HelloJNI.<clinit>(HelloJNI.java:7) Could not find the main class: HelloJNI. Program will exit.
运行java程序:
java HelloJNI
结果见下:
Hello world! Hello world form printString function!!