深入Jar包:Gradle构建可执行jar包与访问jar包中文件夹与文件

前言#

Java的跨平台功能听起来很诱人可口,号称“Write Once,Run Everywhere”,实际上是“Run Once,Debug Everywhere”... 在实际开发过程中还是会遇到各种各样的坑的,刚刚解决了一系列问题,特地写个文章总结一下。

使用Gradle构建Jar包#

感谢万能的Gradle,极大提高了Java开发的生产力~
在Gradle中生成jar包可以使用官方的插件:application 来简单生成Jar包,同时还有多种不同的配置可以自定义,了解详情请参照Gradle官方文档。

我这里使用的是一个叫做 shadow 的Gradle插件,把构建jar包的配置都安排得明明白白了,非常的方便!
官方文档:https://imperceptiblethoughts.com/shadow/configuration/#configuring-output-name

下面是 build.gradle 配置参考:

plugins {
    id 'com.github.johnrengelman.shadow' version '4.0.3'

    // Apply the java plugin to add support for Java
    id 'java'

    // Apply the application plugin to add support for building an application
    id 'application'
}

dependencies {
    implementation 'com.github.jengelman.gradle.plugins:shadow:4.0.3'
}

// Output to build/libs/name.jar
shadowJar {
    baseName = 'name'
    classifier = null
    version = null
}

apply plugin: 'com.github.johnrengelman.shadow'
apply plugin: 'java'

具体的配置要依照项目的实际需要来配置~
设置完 shadow 插件之后,执行 gradle build 就可以在 build/libs/ 文件夹下面生成你的可执行jar包了,超级方便。
需要更多功能可以查看shadow官网文档,写的很清楚。

访问jar包中的资源#

虽然jar包中有各种目录结构,但是jar包本质仍然是一个文件,所以不可以用传统的方法去访问,像 File 类,Class 对象的 getResouce 方法都不行的。
应该使用 ClassLoadergetResourceStream 方法直接获取资源文件的输入流。
例如:

InputStream is=this.getClass().getResourceAsStream("/resource/res.txt");
InputStream is=this.getClass().getClassLoader().getResourceStream("/resource/res.txt");

注意:Class对象和ClassLoader对象的getResourceStream方法也是有不同的,具体的不同可以查看这个笔记:正确获取Java项目资源

访问Jar包中的文件夹#

当jar包中的资源文件很多的时候,不可能一个个输入名字去获取,这也太hack了吧,肯定要用自动化的方式来提高生产力。
事实上,访问jar包中的文件夹是挺麻烦的,不过还是找到了取巧的方法,试了一下还是挺好用的。
(不过最好做一下缓存)

代码如下:

String path = getClass().getProtectionDomain().getCodeSource().getLocation().getPath();
JarFile localJarFile = new JarFile(new File(path));

Enumeration<JarEntry> entries = localJarFile.entries();
while (entries.hasMoreElements()) {
      JarEntry jarEntry = entries.nextElement();
      String innerPath = jarEntry.getName();
      System.out.println(innerPath);
}

使用getClass().getProtectionDomain().getCodeSource().getLocation().getPath(); 来获取当前jar包的路径,如果代码不在jar包中运行的话,获取到的就是当前class文件所在路径。所以在使用之前最好做一下判断,看看程序是否在jar包中运行。

关于JavaFX的Media资源问题#

JavaFX可以播放音乐,但是和其他Image、Font资源不同的是,Media对象的构造函数只能接受一个String参数(即文件URL),所以没办法使用getResourceStream方法把文件输入流传入对象。

我查了一下官网,找到了解决办法,把文件URL换成JarURL就可以了,文档:https://docs.oracle.com/javase/6/docs/api/java/net/JarURLConnection.html。

简单示例:

String path = String.format("jar:file:%s!/%s", jarPath, relativePath);
Media media = new Media(path);

注意:relativePath的形式是 media/hello.wav 这样的。

作者:DealiAxy

出处:https://www.cnblogs.com/deali/p/14377642.html

版权:本作品采用「署名-非商业性使用-相同方式共享 4.0 国际」许可协议进行许可。

微信公众号:「程序设计实验室」
新版StarBlog已经上线,地址:http://blog.deali.cn

posted @   程序设计实验室  阅读(1570)  评论(0编辑  收藏  举报
编辑推荐:
· [.NET]调用本地 Deepseek 模型
· 一个费力不讨好的项目,让我损失了近一半的绩效!
· .NET Core 托管堆内存泄露/CPU异常的常见思路
· PostgreSQL 和 SQL Server 在统计信息维护中的关键差异
· C++代码改造为UTF-8编码问题的总结
阅读排行:
· 【.NET】调用本地 Deepseek 模型
· CSnakes vs Python.NET:高效嵌入与灵活互通的跨语言方案对比
· Plotly.NET 一个为 .NET 打造的强大开源交互式图表库
· DeepSeek “源神”启动!「GitHub 热点速览」
· 上周热点回顾(2.17-2.23)
more_horiz
keyboard_arrow_up dark_mode palette
选择主题
menu
点击右上角即可分享
微信分享提示