动态链接库搜索规则
1、java 中加载 native library
java 中 System.loadLibrary 会调用 Runtime.loadLibary,后者调用 java/lang/ClassLoader.loadLibrary。
在 ClassLoader.loadLibrary 中,会检查系统属性 java.library.path ,在其指定的目录(多个)中搜索 native loadLibrary。
java 启动参数 -Djava.library.path=/path 可以设置搜索目录
2、C 库中加载动态链接库
C 库中 dlopen() 如果参数中不包含“/”,那么用下面的规则搜索动态链接库
a、如果加载链中存在 rpath,则在 rpath 中搜索 (仅 linux ELF)
b、在 LD_LIBRARY_PATH 中搜索
c、在系统 lib 目录中搜索
d、如果加载链中存在 runpath,则在 runpath 中搜索(仅 linux ELF)
两者几乎完全独立,但是 java 是依赖 C 的,所以 java 搜索没有找到时,也会使用 C 库搜索机制。
针对链接库找不到的问题,建议:
a、在构建JNI库时指定 rpath 参数
g++ -shared -rpath . -L xxx -l yyy -o sss.so
b、次优方案是启动时使用 LD_LIBRARY_PATH 环境变量
LD_LIBRARY_PATH=xxxx java -jar aaa.jar