使用jna加载库文件使用以下方式: Native.loadLibrary(dllName, className);java
loadLibrary源码:eclipse
public static Object loadLibrary(String name, Class interfaceClass) {.net
return loadLibrary(name, interfaceClass, Collections.EMPTY_MAP);blog
}ip
即便用dll的名字加载库文件,不能带后缀。get
在不修改代码代码的前提下,那么项目该如何适应dll文件路径的动态变化呢,一方面可以在eclipse中正常运行,导出jar后能在生产环境执行。源码
下面我写了一个方法解决这个问题:it
public synchronized static Object loadDll(String libName, Class<?> className) {
String systemType = System.getProperty("os.name");
String libExtension = (systemType.toLowerCase().indexOf("win") != -1) ? ".dll"
: ".so";
String libFullName = libName + libExtension;
String nativeTempDir = System.getProperty("java.io.tmpdir");
InputStream in = null;
BufferedInputStream reader = null;
FileOutputStream writer = null;
File extractedLibFile = new File(nativeTempDir + File.separator
+ libFullName);
if (!extractedLibFile.exists()) {
try {
in = className.getResourceAsStream("/" + libFullName);
if (in == null)
in = className.getResourceAsStream(libFullName);
reader = new BufferedInputStream(in);
writer = new FileOutputStream(extractedLibFile);
byte[] buffer = new byte[1024];
while (reader.read(buffer) > 0) {
writer.write(buffer);
buffer = new byte[1024];
}
} catch (IOException e) {
e.printStackTrace();
} finally {
if (in != null)
try {
in.close();
} catch (IOException e) {
e.printStackTrace();
}
if (writer != null)
try {
writer.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
String dllName;
if (systemType.toLowerCase().indexOf("win") != -1) {
dllName = extractedLibFile.toString().replace(".dll", "");
} else {
dllName = extractedLibFile.toString().replace(".so", "");
}
return Native.loadLibrary(dllName, className);
}io
加载libHCEHomeCMS,调用例子:class
HCEHomeCMS INSTANCE = (HCEHomeCMS) PropertiesUtils.loadDll("libHCEHomeCMS",HCEHomeCMS.class);
到此,整个项目可使用动态方式加载dll,运行jar也能正常加载,有个问题须要注意下,32位dll文件须要使用32位jdk才能加载。
欢迎指出本文有误的地方,转载请注明原文出处https://my.oschina.net/7001/blog/672173