在java程序中动态设置java.library.path

时间:2023-01-22 09:05:28

最近在做一个项目的过程中需要用到第三方的jar包和动态链接库(dll),其中jar包可以直接引入,问题是在开发的时候dll可以放在System32下,

但是当项目完成后build的时候,这种方式就行不通了,所以必须考虑其他的方式来引用所需的dll文件。

我们知道,在VM参数处通过-Djava.library.path后将加载路径指定到自己的lib目录后,程序可以正常启动

但这种方式显然不够灵活,受限于必须从main函数启动,并且要手动的去指定虚拟机参数。

那么我们现在就需要采取其他的方式,比如在项目的根目录下建一个dll的文件夹,将要用到的dll文件放到此目录下,然后

通过System类的setProperty函数来在代码中动态的改变一下java.library.path的值。

如下:

System.setProperty("java.library.path","%ProjectPath%/dll");

问题是这种方式是行不通的,会报错"no JIntellitype in java.library.path"。

查找原因:

代码中设置不起作用,主要是因为java.library.path只在jvm启动时读取一次,其他情况下的修改不会起作用的。可以参考下面的这个bug:
http://bugs.sun.com/bugdatabase/view_bug.do?bug_id=4280189

原因和ClassLoader的实现有关系。

查看jdk的源码会发现如下实现:

 if (sys_paths == null) {
usr_paths = initializePath("java.library.path");
sys_paths = initializePath("sun.boot.library.path");
}

系统缓存了java.library.path的值,并且一直都会是第一次加载时的值。

有人提到了下面的修改方法,

if(sys_paths == null) {
sys_paths= initializePath("sun.boot.library.path");
}
usr_paths= initializePath("java.library.path");

有问题,就会有人解决问题,google一下,会发现antony_miguel在一篇文章中,使用java的反射机制,完成了对于ClassLoader类中的usr_paths变量的动态修改,

public static void addLibraryDir(String libraryPath) throws IOException {
try {
Field field = ClassLoader.class.getDeclaredField("usr_paths");
field.setAccessible(true);
String[] paths = (String[]) field.get(null);
for (int i = 0; i < paths.length; i++) {
if (libraryPath.equals(paths[i])) {
return;
}
}

String[] tmp = new String[paths.length + 1];
System.arraycopy(paths, 0, tmp, 0, paths.length);
tmp[paths.length] = libraryPath;
field.set(null, tmp);
} catch (IllegalAccessException e) {
throw new IOException(
"Failedto get permissions to set library path");
} catch (NoSuchFieldException e) {
throw new IOException(
"Failedto get field handle to set library path");
}
}


文章也同时指出了这种实现的局限性,和jvm的实现强关联,只要jvm实现不是用的变量usr_paths来保存java.library.path的值,这个方法就不能用了。
但是只要知道源代码,小小的改动就应该可以实现了。

 

通过调用上面的方法来将dll文件目录加入java.library.path路径下,然后使用System.loadLibrary("glpk_4_55")这样的方式加载dll文件即可。

 

over~