System.load(AAA)这个加载本地库的方法,会自动到System.getProperty(“java.library.path”)的地址里面去寻找AAA.dll
java.library.path里面包括一组地址,包含系统变量PATH,它必然包括%javahome%/bin,所以当我们把dll扔到%javahome%/bin下面的时候,则可以直接加载成功.
但在有的情况下,我们希望dll不污染虚拟机,而是放在项目里面.那这就是动态加载库.
我们可以尝试修改java.library.path
String currentPath=...
System.setProperty("java.library.path", currentPath+";"+System.getProperty("java.library.path"));
但这没有什么用.
原因在于java.library.path会在jvm启动时(此时sys_paths 属性为null)才加载.
如下图ClassLoader里的源码,可以看到当sys_paths 属性是null,加载了java.library.path
static void loadLibrary(Class<?> fromClass, String name,
boolean isAbsolute) {
ClassLoader loader =
(fromClass == null) ? null : fromClass.getClassLoader();
if (sys_paths == null) {
usr_paths = initializePath("java.library.path");
sys_paths = initializePath("sun.boot.library.path");
}
if (isAbsolute) {
if (loadLibrary0(fromClass, new File(name))) {
return;
}
throw new UnsatisfiedLinkError("Can't load library: " + name);
}
...
}
那么我们可以取巧,在设置java.library.path后设置sys_paths 为null
Field field = ClassLoader.class.getDeclaredField("sys_paths");
field.setAccessible(true);
field.set(null, null);
这样就有效果了.