在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!\n");
return ;
} JNIEXPORT void JNICALL Java_HelloJNI_printString(JNIEnv *env, jobject obj, jstring string)
{
const char * str = (*env)->GetStringUTFChars(env,string,0);
printf("%s!\n",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!!