使用 javah 生成头文件后, 编写c代码来实现其中声明的函数, 本文主要解决以下问题:
(1) 如何生成动态链接库文件
使用如下格式的 gcc 命令可以将 C文件 编译为 .so 文件, 对于其依赖则稍微复杂一点, 需要使用各种参数.
cc -shared -o <file name>.so <file.c...>
对于 Linux 而言, 编写 JNI 本机程序需要包含 $JDK_HOME/include 和 $JDK_HOME/include/linux, 其中 $JDK_HOME 是 JDK 的安装目录, 类似 /usr/lib/jvm/java-8-openjdk-amd64
-l参数和-L参数
-l参数就是用来指定程序要链接的库,-l参数紧接着就是库名,那么库名跟真正的库文件名有什么关系呢?就拿数学库来说,他的库名是m,他的库文件名是libm.so,很容易看出,把库文件名的头lib和尾.so去掉就是库名了。
-L参数跟着的是库文件所在的目录名。再比如我们把libtest.so放在/aaa/bbb/ccc目录下,那链接参数就是-L/aaa/bbb/ccc -ltest另外,大部分libxxxx.so只是一个链接
-include和-I参数
-include用来包含头文件,但一般情况下包含头文件都在源码里用#include xxxxxx实现,-include参数很少用。-I参数是用来指定头文件目录,/usr/include目录一般是不用指定的,gcc知道去那里找,但是如果头文件不在/usr/include里我们就要用-I参数指定了,比如头文件放在/myinclude目录里,那编译命令行就要加上-I/myinclude参数了,如果不加你会得到一个"xxxx.h: No such file or directory"的错误。-I参数可以用相对路径,比如头文件在当前目录,可以用-I.来指定
(2) 如何使用 .so 文件
通过 Java 程序调用 .so 中的函数, 首先需要使用 System.load(String) 或 System.loadLibrary(String) 将 .so 文件加载.
其中 load(String path) 方法需要 .so 文件的绝对路径, 而 loadLibrary(String name) 方法则从 Java 运行时变量 java.library.path 中声明的路径中查找 libname.so 文件然后加载, java.library.path 是多个变量的叠加, 可以通过 Linux 用户变量 PATH 和 LD_LIBRARY_PATH 修改.