通过JNI调用C程序实现"Hello World!"输出,掌握此方法可以很方便地将Java函数用C/C++实现。
确定安装成功Java,本机上因为工程需要,已经安装几个jdk版本,执行java -version
root@wang:~/jni# java -version
openjdk version "1.8.0_91"
OpenJDK Runtime Environment (build 1.8.0_91-8u91-b14-0ubuntu4~14.04-b14)
OpenJDK 64-Bit Server VM (build 25.91-b14, mixed mode)
可以看到使用OpenJDK 8的版本。
编写可以HelloWorld.java程序,有一个native方法声明为DisplayHello.
public class HelloWorld {
static {
System.loadLibrary("Hello");
}
public native void DisplayHello();
public static void main(String[] args) {
new HelloWorld().DisplayHello();
}
}
执行javac HelloWorld.java生成HelloWorld.class文件
执行javah -jni HelloWorld会在当前目录下生成HelloWorld.h文件
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class HelloWorld */
#ifndef _Included_HelloWorld
#define _Included_HelloWorld
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: HelloWorld
* Method: DisplayHello
* Signature: ()V
*/
JNIEXPORT void JNICALL Java_HelloWorld_DisplayHello
(JNIEnv *, jobject);
#ifdef __cplusplus
}
#endif
#endif
此文件为javah自动生成的,不需要修改,可以看到里面包含一个Java_HelloWorld_DisplayHello的函数声明。
编写imp_hello.c程序,此程序实现刚刚声明的函数。
#include <jni.h>
#include "HelloWorld.h"
#include <stdio.h>
JNIEXPORT void JNICALL Java_HelloWorld_DisplayHello(JNIEnv * env, jobject obj) {
printf("From imp_hello.c:");
printf(" Hello world ! \n");
return;
}
此文件包括两个头文件jni.h和生成的HelloWorld.h,实现函数中就打印一句话From imp_hello.c: Hello world !
生成动态库命令:
gcc -shared -fPIC -I /usr/lib/jvm/java-8-openjdk-amd64/include/ -I /usr/lib/jvm/java-8-openjdk-amd64/include/Linux/ imp_hello.c -o libHello.so
不同的机器可能jdk的路径不相同,可以通过上面的命令查看使用的jdk版本,包含两个文件路径,一个是jni.h文件路径,一个是jni_md.h的文件路径。如果碰到其它的错误,可以根据提示解决,比如本机上如果不包含-fPIC,就会提示出错:
root@wang:~/jni# gcc -shared -I /usr/lib/jvm/java-8-openjdk-amd64/include/ -I /usr/lib/jvm/java-8-openjdk-amd64/include/linux/ imp_hello.c -o libHello.so
/usr/bin/ld: /tmp/ccXqfTfA.o: relocation R_X86_64_32 against `.rodata' can not be used when making a shared object; recompile with -fPIC
/tmp/ccXqfTfA.o: error adding symbols: Bad value
collect2: error: ld returned 1 exit status
提示中已经给出解决方法,recompile with -fPIC,如果碰到其它的错误也可以上网查询相关的问题。
编译完成之后当前目录下生成libHello.so库。
root@wang:~/jni# ls
HelloWorld.class HelloWorld.h HelloWorld.java imp_hello.c libHello.so
最后执行:
root@wang:~/jni# java -Djava.library.path=. HelloWorld
From imp_hello.c: Hello world !
此时Java程序已经成功调用C程序,如果是C++程序则可以将imp_hello.c命名为imp_hello.cpp,将gcc改成g++,其它步骤和此相同。