JNI是Java Native Interface的缩写,从Java1.1开始,JNI标准成为java平台的一部分,它允许Java代码和其他语言写的代码进行交互。JNI一开始是为了本地已编译语言,尤其是C和C++而设计的,但是它并不妨碍你使用其他语言,只要调用约定受支持就可以了。
目前java与dll交互的技术主要有3种:jni,jawin和jacob。Jni(Java Native Interface)是sun提供的java与系统中的原生方法交互的技术(在windowslinux系统中,实现java与native method互调)。目前只能由c/c++实现。后两个都是sourceforge上的开源项目,同时也都是基于jni技术的windows系统上的一个应用库。Jacob(Java-Com Bridge)提供了java程序调用microsoft的com对象中的方法的能力。而除了com对象外,jawin(Java/Win32 integration project)还可以win32-dll动态链接库中的方法。就功能而言:jni >> jawin>jacob,就易用性而言,正好相反:jacob>jawin>>jni。
Jni程序开发的一般操作步骤如下:
- 编写java类声明native方法;
- 用javah生成c/c++原生函数的头文件;
- 编写c/c++代码实现原生函数并编译成库(windows是dll,linux是so);
- 通过System.loadLibrary()或System.load()加载生成的库,或则给虚拟机传参(java.library.path)指定库的路径;
- java调用native方法进行业务处理;
下面我们按部就班地进行操作(windows下),编写java类声明native方法
项目结构如下如下:
App.java代码如下:
package net.oseye.JniDemo; public class App { public static void main( String[] args ) { //调用native方法 new Hello().sayHello(); } } class Hello{ static{ System.loadLibrary("libhello"); } /* * 声明native方法 */ public native void sayHello(); }
编译后会生成App.class和Hello.class。
用javah生成c/c++原生函数的头文件
使用命令
D:workspace4jeeJniDemo argetclasses>javah -jni net.oseye.JniDemo.Hello
生成c/c++头文件 net_oseye_JniDemo_Hello.h:
net_oseye_JniDemo_Hello.h代码:
/* DO NOT EDIT THIS FILE - it is machine generated */ #include <jni.h> /* Header for class net_oseye_JniDemo_Hello */ #ifndef _Included_net_oseye_JniDemo_Hello #define _Included_net_oseye_JniDemo_Hello #ifdef __cplusplus extern "C" { #endif /* * Class: net_oseye_JniDemo_Hello * Method: sayHello * Signature: ()V */ JNIEXPORT void JNICALL Java_net_oseye_JniDemo_Hello_sayHello (JNIEnv *, jobject); #ifdef __cplusplus } #endif #endif
使用c/c++实现native方法
在与net_oseye_JniDemo_Hello.h的目录下建立hello.cpp,代码:
#include <stdio.h> #include "net_oseye_JniDemo_Hello.h" JNIEXPORT void JNICALL Java_net_oseye_JniDemo_Hello_sayHello (JNIEnv *, jobject) { printf("Hello, world "); }
使用gcc编译成dll,命令:
D:workspace4jeeJniDemo argetclasses>gcc -shared -Wl,--kill-at -I "d:Program FilesJavajdk1.7.0_05include" hello.cpp -o libhello.dll
此时我的D:workspace4jeeJniDemo argetclasses路径结构如下:
执行Java程序
D:workspace4jeeJniDemo argetclasses>java -Djava.library.path=. net.oseye.JniDemo.App
输出
Hello, world
以备不时之需的PS:
- linux下有非常好用的c/c++编译器gcc,windows下也有移植,貌似大家比较喜欢MinGW,但需要在线安装,因此需要访问公网权限。我使用了TDM-GCC,它可离线安装,它结合了 GCC 工具集中最新的稳定发行版本,包括了自由并开源的 MinGW 或 MinGW-w64 的运行时 APIs,以此创建一个 LIBRE 来替代微软的编译器及其平台 SDK。GCC简单使用教程可查看百度文库。
-
JNI基本类型
Java类型本地类型描述booleanjbooleanC/C++8位整型bytejbyteC/C++带符号的8位整型charjcharC/C++无符号的16位整型shortjshortC/C++带符号的16位整型intjintC/C++带符号的32位整型longjlongC/C++带符号的64位整型floatjfloatC/C++32位浮点型doublejdoubleC/C++64位浮点型Objectjobject任何Java对象,或者没有对应java类型的对象ClassjclassClass对象Stringjstring字符串对象Object[]jobjectArray任何对象的数组boolean[]jbooleanArray布尔型数组byte[]jbyteArray比特型数组char[]jcharArray字符型数组short[]jshortArray短整型数组int[]jintArray整型数组long[]jlongArray长整型数组float[]jfloatArray浮点型数组double[]jdoubleArray双浮点型数组
-
error: parameter name omitted
如果你用c实现javah生成的头文件,可能会遇到这个这个问题,这是由于C与C++的细微区别造成的:- 在函数声明中:无论是C还是在C++,都可以省略形式参数名。但是,通常都不建议省略形式参数名.
-
在函数实现中:
- 当需要使用形式参数的时候,显然,必须给形式参数命名。
-
当不需要使用形式参数的时候,C与C++有微小差异:
C不能省略形式参数名, 即使不使用。
C++可以省略形式参数名,如果不使用。并且在C++中,如果给不使用的形式参数命名,可能会得到一个警告。
-
java.lang.Unsatisfie.lang.UnsatisfiedLinkError no XXXXX in java.library.path
报这个异常主要是找不到你的库(dll或so)文件,你可以使用-Djava.library.path指定库文件地址,或者你通过System.getProperty("java.library.path")获取默认java.library.path地址,把库文件拷贝到里面去。 - java.lang.UnsatisfiedLinkError: XXXclass.XXXmethod()
这个错误是 在这个dll里找不到方法的声明,网上说是@符号的问题 ,主要有三种解决方法:- 第1种方法:
gcc -Wl,--kill-at -shared -o jnihello.dll Native.c
- 第2种方法:
gcc -Wl,--add-stdcall-alias -shared -o jnihello22.dll Native.c
- 第3种方法: 在你的本地方法的头文件中中的函数前面加上下划线,比如以前是
JNIEXPORT void JNICALL Java_net_oseye_JniDemo_Hello_sayHello(JNIEnv *, jobject);
JNIEXPORT void JNICALL _Java_net_oseye_JniDemo_Hello_sayHello(JNIEnv *, jobject);
- 第1种方法: