本文介绍java使用jni调用c、c++编写生成的dll文件或so文件中的接口方法实现过程。
1. 编写java含有本地方法的测试类。
package com.gh;
publicclass JniGet {
publicnative StringgetSomething();
publicstaticvoid main(String[]args){
Stringsomething =new JniGet().getSomething();
System.out.println(something);
}
}
2. 生成.h文件。
命令行进入java项目bin目录下运行:javah com.gh.JniGet,会在bin目录下生成com_gh_JniGet.h文件。
3. 在vs中文件-新建-Win32项目,然后选DLL,完成dll项目创建。将生成的com_gh_JniGet.h加入到项目中并编写jniget.c
com_gh_JniGet.h内容如下:
/*DO NOT EDIT THIS FILE - it is machine generated */
#include<jni.h>
/*Header for class com_gh_JniGet */
#ifndef_Included_com_gh_JniGet
#define_Included_com_gh_JniGet
#ifdef__cplusplus
extern"C" {
#endif
/*
* Class: com_gh_JniGet
* Method: getSomething
* Signature: ()Ljava/lang/String;
*/
JNIEXPORTjstringJNICALLJava_com_gh_JniGet_getSomething
(JNIEnv *,jobject);
#ifdef__cplusplus
}
#endif
#endif
jnitest.c内容如下:
//jniget.cpp : 定义DLL 应用程序的导出函数。
//
#include<stdio.h>
#include"stdafx.h"
#include"com_gh_JniGet.h"
JNIEXPORTjstringJNICALLJava_com_gh_JniGet_getSomething(JNIEnv * a,jobjectb){
returna->NewStringUTF("1de816c5-c618-4528-bd6d-1dac14fd3478");
}
4. 生成jniget.dll文件,并将dll文件复制到java.library.path对应的目录中,可以通过System.out.println(System.getProperty("java.library.path"));查看有哪些目录。
5. 测试dll文件中方法
在JniGet类中加入如下代码
static{
System.loadLibrary("jniget");
}
运行JniGet,至此便可以从dll文件中接口方法获取数据。
注意点
A.“#include <jni.h>” 找不到jni.h的引用。解决方法,将jdk安装目录下的include文件夹中的“jni.h”,及当前文件夹下win32文件夹中“jni_md.h”、“jawt_md.h”,共3个文件,拷入VS2010安装目录下的“/VC/include/”文件夹中。
B.生成的是32位dll在64位jvm下运行报错
Exception in thread "main"java.lang.UnsatisfiedLinkError: C:\ProgramFiles\Java\jdk1.7.0_67\bin\jniget.dll: Can't load IA 32-bit .dll on a AMD64-bit platform
解决方法:(1)右键解决方案->属性,将平台选为X64,若无此选项,则点击配置管理器,选择X64平台,若无则新建一个X64平台并选中。(2)项目----属性-----链接器-----高级-----目标计算机,改为 X64。
6. linux系统下生成so文件
将com_gh_JniGet.h jniget.cpp放到同一个目录下编译,cpp中去掉#include"stdafx.h"
g++ -I /usr/java/jdk1.7.0_03/include -I/usr/java/jdk1.7.0_03/include/linux -fPIC -shared -o libjniget.so jniget.cpp
其中 -I后面是java的include文件夹地址,请根据您具体的java版本以及安装路径作相应改变;
-f后面的PIC表示生成的库中符号是与位置无关的(PIC就是Position Independent Code)。
-shared表示共享,共享库后缀名.so可以认为是sharedobject的简称;
-o libIntArray.so,可以理解为编译后输出libIntArray.so库。
注意项:
A.使用g++不是gcc
B.可能出现Exception inthread "main" java.lang.UnsatisfiedLinkError: no jniget injava.library.path
解决方法有两个:
1、临时指定共享库libIntArray.so的路径。
export LD_LIBRARY_PATH=.:$LD_LIBRARY_PATH 注:该方法只在当前会话窗口有效,切换到另外一个终端窗口,则需要重新指定共享库路径。也可直接在环境变量中加入:LD_LIBRARY_PATH=.:$ORACLE_HOME/lib:/usr/lib64:/usr/lib:/usr/local/lib64:/usr/local/lib 命令行中使用:source ~/.bash_profile 使当前增加的环境变量生效
2、运行时加上参数指定共享库libIntArray.so的路径。
java -Djava.library.path=. IntArray 注:-D:设置Java平台的系统属性。 此时JavaVM可以在当前位置找到该库。
C.共享库命名格式一定要是libxxx.so,xxx为System.loadLibrary(“xxx”);。