在Java开发时,我们有时候会接触到很多本地库,这样在对项目打包的时候我们不得不面临一个选择:要么将库文件与包好的jar文件放在一起;要么将库文件包入jar。
将一个不大的项目包成一个jar有诸多发布优势,本次将分享一个将JNI包入jar的方法。
[实现思路]
将JNI库(dll、so等)包入jar后,我们无法通过路径来访问它们,而库的读取依赖一个java.library.path下对应名称的外部库文件,我们仅仅需要在调用JNI前将其由jar包释放出来,这类似于文件的拷贝过程。
[部署位置的选取]
java.library.path并不是一个固定的位置,使用下面代码可以打印出来:
System.out.println(System.getProperty("java.library.path"));
例如在本人的计算机上,结果是这样的:
D:\Program Files (x86)\Java\jre7\bin;C:\windows\Sun\Java\bin;C:\windows\system32;C:\windows;D:/ProgramFiles (x86)/Java/jre7/bin/client;D:/Program Files(x86)/Java/jre7/bin;D:/Program Files (x86)/Java/jre7/lib/i386;C:\Program Files(x86)\NVIDIA Corporation\PhysX\Common;C:\windows\system32;C:\windows;C:\windows\System32\Wbem;C:\windows\System32\WindowsPowerShell\v1.0\;E:\Develop\jdk1.7.0_71\bin;E:\Develop\Git\cmd;E:\Develop\Git\bin;E:\Develop\apache-maven-3.2.1\bin;E:\eclipse-java-luna-SR1-win32\eclipse;;.
绝对路径会由于不同系统而变化,因而这里选择了”.”这个相对路径,也就是项目目录下。
[JNI部署类]
import java.io.File; import java.io.FileOutputStream; import java.io.InputStream; import java.util.Arrays; import java.util.LinkedList; import java.util.List; public class JNIDevelopment { byte[] cache; List<String> sources; public JNIDevelopment(){ cache = new byte[1024]; sources = new LinkedList<String>(); //这里加入本地库文件名,也可以稍加修改读取一个外部xml或properties sources.add("luajava-1.1.dll"); sources.add("libluajava-1.1.so"); } private Boolean sourceExist(String sourceName){ File f = new File("." + File.separator + sourceName); return f.exists(); } public void doDefaultDevelopment(){ for (String s:sources){ doDevelopment(s); } } public Boolean doDevelopment(String sourceName){ if(sourceExist(sourceName)){ return true; } else{ try{ File f = new File("." + File.separator + sourceName); if(!f.exists()){ f.createNewFile(); System.out.println("[JNIDEV]:DEFAULT JNI INITION:"+sourceName); } FileOutputStream os = new FileOutputStream(f); InputStream is = getClass().getResourceAsStream(sourceName); if(is == null){ os.close(); return false; } Arrays.fill(cache,(byte)0); int realRead = is.read(cache); while(realRead != -1){ os.write(cache, 0, realRead); realRead = is.read(cache); } os.close(); } catch(Exception e){ System.out.println("[JNIDEV]:ERROR IN COPY JNI LIB!"); e.printStackTrace(); return false; } } return true; } public static void main(String[] args) { JNIDevelopment deve = new JNIDevelopment(); deve.doDefaultDevelopment(); try{ System.loadLibrary("luajava-1.1"); System.out.println("本地库加载成功"); } catch(UnsatisfiedLinkError e){ System.out.println("本地库加载失败"); } } }
之后我们将本地库放置在与该类同包下:
[运行结果]
[JNIDEV]:DEFAULT JNI INITION:luajava-1.1.dll
[JNIDEV]:DEFAULT JNI INITION:libluajava-1.1.so
本地库加载成功
总结
以上就是本文关于浅谈将JNI库打包入jar文件的全部内容,希望对大家有所帮助。感兴趣的朋友可以参阅本站其他Java相关专题。如有不足之处,欢迎留言指出。感谢朋友们对本站的支持!
问题内容: 我在eclipse项目中创建了一个derby嵌入式数据库,它在eclipse上运行良好,但是当将该项目打包到Runnable jar文件中时,它无法连接数据库。 我已经完成了与此视频类似的操作 http://vinayakgarg.wordpress.com/2012/03/07/packaging-java- application-with-apache-derby-as-jar-
本文向大家介绍浅谈Linux的库文件,包括了浅谈Linux的库文件的使用技巧和注意事项,需要的朋友参考一下 最近在Linux下使用第三方库Protobuf时,遇到一个问题:可执行程序在运行时报错:“error while loading shared libraries: libprotobuf.so.7: cannot open shared object file: No such file
本文向大家介绍浅谈React + Webpack 构建打包优化,包括了浅谈React + Webpack 构建打包优化的使用技巧和注意事项,需要的朋友参考一下 本文介绍了React + Webpack 构建打包优化,分享给大家,具体如下: 使用 babel-react-optimize 对 React 代码进行优化 检查没有使用的库,去除 import 引用 按需打包所用的类库,比如 lodash
问题内容: 我想知道使用可运行的jar文件创建将库从eclipse提取和打包成jar文件之间的区别。 如果我的程序(可运行的jar)使用需要这些外部库(jar)的其他类,我应该选择什么? 问题答案: 如果要将jar放入生成的jar文件中,则可以使用打包方法。例如,如果您使用的是Apache库或其他一些第三方的jar,则可能需要将这些jar保存在生成的jar中。在这种情况下,请使用包装。 “将所
15.2.2.打包库文件 如果应用程序需要加载某个本地库,那就必须保证系统能够找到它。在Linux下,它们通常都是通过LD_LIBRARY_PATH中定义的路径找到。在Android中,这个环境变量中只包含一个路径,即system/lib。这里存在一个问题,那就是/system分区一般都是只读的,应用程序无法将自己的本地库安装到/system/lib。 为解决这一问题,NDK使用了这样的机制,那就
本文向大家介绍浅谈vue.js导入css库(elementUi)的方法,包括了浅谈vue.js导入css库(elementUi)的方法的使用技巧和注意事项,需要的朋友参考一下 1.安装以下模块,让webpack可以解析css文件 2.安装elementUi模块 3.在webpack.base.conf.js中添加配置 4.然后在 main.js 引入并注册 以上这篇浅谈vue.js导入css库(e