您的位置:首页 > 运维架构 > Linux

Java在linux下调用C/C++生成的so文件

2018-03-05 17:46 393 查看
Java在linux下调用C/C++生成的so文件

CplusUtil.java是java web工程中的一个工具类

CplusUtil.java 内容如下:
package cn.undoner.utils;

/**
* Created by ${<A HREF="mailto:undoner@gmail.com">undoner</A>} on 16-2-25.
*/
public class CplusUtil {

public native void sayHello();

static{
System.out.println(System.getProperty("java.library.path"));
System.load("/usr/lib/jvm/java/jre/lib/amd64/server/MyJni.so");
}

public static void main(String[] args){
CplusUtil h = new CplusUtil();
h.sayHello();
}
}

生成相应class文件

可通过IDE工具(Eclipse/IntellJ/javac等)编译整个工程,生成相应class文件 本文路经为:
/工程名/target/classes/cn/undoner/utils/CplusUtil.class

添加.h文件

为utils所需调用的c函数生成相应.h头文件注意:在有包名情况下,生成.h头文件要记得带上包名路径
javah -classpath classes -jni cn.undoner.utils.CplusUtil
生成文件:
cn_undoner_utils_CplusUtil.h
 内容如下:
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class cn_undoner_utils_CplusUtil */

#ifndef _Included_cn_undoner_utils_CplusUtil
#define _Included_cn_undoner_utils_CplusUtil
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class:     cn_undoner_utils_CplusUtil
* Method:    sayHello
* Signature: ()V
*/
JNIEXPORT void JNICALL Java_cn_undoner_utils_CplusUtil_sayHello
(JNIEnv *, jobject);

#ifdef __cplusplus
}
#endif
#endif

编写c函数

文件:
MyJni.c
 内容如下:
#include <jni.h>
#include "cn_undoner_utils_CplusUtil.h"
#include <stdio.h>

JNIEXPORT void JNICALL Java_cn_undoner_utils_CplusUtil_sayHello(JNIEnv *env,jobject obj){
printf("Hello JNI");
return;
}

编译.c文件

将.c文件编译成.o,再重新转换成.so或者.dll文件
gcc -fPIC -D_REENTRANT -I /usr/lib/jvm/java/include -I /usr/lib/jvm/java/include/linux -c MyJni.c
生成文件:
MyJni.o

将.o文件编译成.so文件

gcc -shared MyJni.o -o MyJni.so
生成文件:
MyJni.so

本文MyJni.so路径为:

/usr/lib/jvm/java/jre/lib/amd64/server/MyJni.so

执行CplusUtil.class

在ide中运行CplusUtil或者直接java命令执行:
java cn.undoner.utils.CplusUtil

结果

/usr/lib/jvm/java/jre/lib/amd64/server:
/usr/lib/jvm/java/jre/lib/amd64:
/usr/lib/jvm/java/jre/../lib/amd64:
/home/vobile/java_tool/idea-IU-141.178.9/bin::
/usr/java/packages/lib/amd64:
/usr/lib64:
/lib64:
/lib:
/usr/lib
Hello JNI
注:
System.getProperty("java.library.path")
输出以下信息为本机的lib路径:
/usr/lib/jvm/java/jre/lib/amd64/server:
/usr/lib/jvm/java/jre/lib/amd64:
/usr/lib/jvm/java/jre/../lib/amd64:
/home/vobile/java_tool/idea-IU-141.178.9/bin::
/usr/java/packages/lib/amd64:
/usr/lib64:/lib64:
/lib:/usr/lib
简化调用linux下Java调用so文件。需要注意以下几点:在linux下调用的so文件的格式必须是linux的,而不是x86或其他。
linux打包的so文件的位数必须和本机JVM的位数一样,都是32位或都是64位,JVM位数可以通过java -version查看。如果位数不一致就会报以下错误:wrong ELF class: ELFCLASS64 (Possible cause: architecture word width mismatch) http://www.yanzuoguang.com:9000/upload/2017/11/sn2q6bsk58hb2rh9cb6mvmu3bd.png so包path要配置好。如果没配置好就会报以下错误:java.lang.UnsatisfiedLinkError: no *** in java.library.path http://www.yanzuoguang.com:9000/upload/2017/11/6gshbosr44h8uohu2tidquf120.png 可以用 System.out.println("java.library.path:"+System.getProperty("java.library.path")) 来看path是什么。出现错误的原因是java path中找不到so包。解决办法有2个。 a. 把so包cp到对应的path中。 b. 在命令行中定义so包的path。
export LD_LIBRARY_PATH=/**/*/** (so所在的目录)
只在当前shell下有效... 4. 开始使用了,在JAVA开头中load一下。
System.loadLibrary("stockocr"); 
so文件的全名是libstockocr.so。注意linux下load时需要去掉lib和so。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  Linux Java jni so