linux环境java如何调用so文件

linux环境java如何调用so文件,第1张

用JNI实现实例:创建HelloWorld.javaclass HelloWorld{private native void print()public staticvoid main(String[] args){new HelloWorld().print()}static{System.loadLibrary("HelloWorld")}}注意print方法的声明,关键字native表明该方法是一个原生代码实现的。另外注意static代码段的System.loadLibrary调用,这段代码表示在程序加载的时候,自动加载libHelloWorld.so库。编译HelloWorld.java在命令行中运行如下命令:javac HelloWorld.java在当前文件夹编译生成HelloWorld.class。生成HelloWorld.h在命令行中运行如下命令:javah -jni HelloWorld在当前文件夹中会生成HelloWorld.h。打开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 __cplusplusextern "C" {#endif/* * Class: HelloWorld * Method:print * Signature: ()V */JNIEXPORT void JNICALL Java_HelloWorld_print(JNIEnv *, jobject)#ifdef __cplusplus}#endif#endif该文件中包含了一个函数Java_HelloWorld_print的声明。这里面包含两个参数,非常重要,后面讲实现的时候会讲到。实现HelloWorld.c创建HelloWorld.c文件输入如下的代码:#include <jni.h>#include <stdio.h>#include "HelloWorld.h"JNIEXPORT void JNICALLJava_HelloWorld_print(JNIEnv *env, jobject obj){printf("Hello World!\n")}注意必须要包含jni.h头文件,该文件中定义了JNI用到的各种类型,宏定义等。另外需要注意Java_HelloWorld_print的两个参数,本例比较简单,不需要用到这两个参数。但是这两个参数在JNI中非常重要。env代表java虚拟机环境,Java传过来的参数和c有很大的不同,需要调用JVM提供的接口来转换成C类型的,就是通过调用env方法来完成转换的。obj代表调用的对象,相当于c++的this。当c函数需要改变调用对象成员变量时,可以通过 *** 作这个对象来完成。编译生成libHelloWorld.so在Linux下执行如下命令来完成编译工作:cc -I/usr/lib/jvm/java-6-sun/include/linux/-I/usr/lib/jvm/java-6-sun/include/-fPIC -shared -o libHelloWorld.so HelloWorld.c在当前目录生成libHelloWorld.so。注意一定需要包含Java的include目录(请根据自己系统环境设定),因为Helloworld.c中包含了jni.h。另外一个值得注意的是在HelloWorld.java中我们LoadLibrary方法加载的是“HelloWorld”,可我们生成的Library却是libHelloWorld。这是Linux的链接规定的,一个库的必须要是:lib+库名+.so。链接的时候只需要提供库名就可以了。运行Java程序HelloWorld大功告成最后一步,验证前面的成果的时刻到了:java HelloWorld如果你这步发生问题,如果这步你收到java.lang.UnsatisfiedLinkError异常,可以通过如下方式指明共享库的路径:java -Djava.library.path='.' HelloWorld当然还有其他的方式可以指明路径请参考《在Linux平台下使用JNI》。我们可以看到久违的“Hello world!”输出了。

在windows环境下生成的动态库是 dll

在linux下生成so

在mac下生成的是dylib

所以对于Android的开发者来说,还是要用ndk的工具

把在idea生成的.h文件粘贴到clion的项目中就可以了,

如果找不到jni.h可以在CMakeList.txt中添加对jni.h文件的引用

如下:

后面三行根据jdk的实际位置填写,作用是导入jni相关的头文件;其他的 *** 作系统可能不完全一致

复制.h头文件 com_shenby_jni_JniExample.h

实现该文件定义的函数 com_shenby_jni_JniExample.c

在CMakeList.txt中添加生成动态共项库

add_library:生成一个库,add_executable:生成一个可执行文件

hello:是生成共享库的名字,前面会自动加上lib前缀,如这里windows生成的是 libhello.dll

SHARED:库的类型为动态,windows上生成.dll,而STATIC 则是生成静态库,windows生成.a文件

com_shenby_jni_JniExample.c :c源文件,多个文件就添加在后面,切记不能添加上面的com_shenby_jni_JniExample.h

文件,否则生成的库中的函数为空

结果如下

就可以把该dll文件粘贴到idea的项目根目录中运行了


欢迎分享,转载请注明来源:内存溢出

原文地址: https://outofmemory.cn/yw/6271614.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-03-19
下一篇 2023-03-19

发表评论

登录后才能评论

评论列表(0条)

保存