java打包jar后读取依赖jar包中的文件资源,支持读取jlink打包的模块镜像中读取

java项目通常会打包成jar,jar里面依赖第三方的jar,那如何在我们的程序中读取第三方jar中的资源呢,当然还有一种情况,是从新版jdk通过jlink打包生成的镜像中获取第三方资源

这里提供两种打包方式的获取方法

  1. 普通打包的jar,例如springboot的jar
  2. 模块化打包的程序,例如jlink打包的镜像
java 复制代码
		try {
			//普通jar包中获取
			//jar:file:///jar包具体位置!/包中资源具体位置
//			URL url = new URL("jar:file:///D:/maven/repository/org/bytedeco/ffmpeg/6.1.1-1.5.10/ffmpeg-6.1.1-1.5.10-windows-x86_64-gpl.jar!/org/bytedeco/ffmpeg/windows-x86_64-gpl/ffmpeg.exe");
			
			//模块化中获取
			//jrt:/模块名/包中资源具体位置
			URL url = new URL("jrt:/org.bytedeco.ffmpeg.windows.x86_64.gpl/org/bytedeco/ffmpeg/windows-x86_64-gpl/ffmpeg.exe");
			InputStream openStream = url.openStream();
			System.out.println(openStream.available());
			
		} catch (java.lang.Exception e) {
			e.printStackTrace();
		}
		

模块化打包获取第三方jar中资源的另外一种方式

新的jdk支持模块化打包,此时是以jrt协议访问的,同时jdk新增了FileSystem类,可以用来访问第三方jar中的资源

java 复制代码
		FileSystem fs = FileSystems.getFileSystem(URI.create("jrt:/"));
        //fs.getPath("modules/模块名", "包中资源具体的位置");
        Path imagePath = fs.getPath("modules/org.bytedeco.ffmpeg.windows.x86_64.gpl", "org/bytedeco/ffmpeg/windows-x86_64-gpl/ffmpeg.exe");
        byte[] bytes = null;
		try {
			bytes = Files.readAllBytes(imagePath);
		} catch (IOException e) {
			e.printStackTrace();
		}
        System.out.println(bytes.length);

通用获取资源方法

java 复制代码
try {
			String jarPathName = "ffmpeg-6.1.1-1.5.10-windows-x86_64-gpl";
			String jrtModuleName = "org.bytedeco.ffmpeg.windows.x86_64.gpl";
			
			ClassLoader systemClassLoader = App.class.getClassLoader();
			Enumeration<URL> systemResources = systemClassLoader.getResources("META-INF/MANIFEST.MF");
			Iterator<URL> iterator = systemResources.asIterator();
			while (iterator.hasNext()) {
				URL next = iterator.next();
				String path = next.toString();
				
				//ffmpeg-6.1.1-1.5.10-windows-x86_64-gpl是普通jar的路径格式,直接以文件名表示
				//org.bytedeco.ffmpeg.windows.x86_64.gpl是jlink打包镜像中以jrt:协议的路径格式,他其实是路径中的模块名
				//通过动态获取App的所有类和第三方依赖jar,进行过滤,再进行拼接第三方jar中资源的具体路径
				if (path.indexOf(jarPathName) >= 0 || path.indexOf(jrtModuleName) >= 0) {
					String replace = path.replace("META-INF/MANIFEST.MF", "");
					
					//jar:file:///jar包具体位置!/包中资源具体位置
					//jrt:/模块名/包中资源具体位置
					replace = replace + "org/bytedeco/ffmpeg/windows-x86_64-gpl/ffmpeg.exe";
					
					System.out.println(replace);
					
					URL url = new URL(replace);
					InputStream openStream = url.openStream();
					System.out.println("ffmpeg:"+openStream.available());
				}

			}
		} catch (java.lang.Exception e) {
			e.printStackTrace();
		}

通过Main方法类的getClassLoader可以获取当前程序所有依赖类和第三方jar,然后通过过滤得到路径,就能获取jar包或者jlink模块镜像中的资源了

相关推荐
超级大只老咪1 小时前
数组相邻元素比较的循环条件(Java竞赛考点)
java
小浣熊熊熊熊熊熊熊丶1 小时前
《Effective Java》第25条:限制源文件为单个顶级类
java·开发语言·effective java
毕设源码-钟学长2 小时前
【开题答辩全过程】以 公交管理系统为例,包含答辩的问题和答案
java·eclipse
啃火龙果的兔子2 小时前
JDK 安装配置
java·开发语言
星哥说事2 小时前
应用程序监控:Java 与 Web 应用的实践
java·开发语言
派大鑫wink2 小时前
【JAVA学习日志】SpringBoot 参数配置:从基础到实战,解锁灵活配置新姿势
java·spring boot·后端
xUxIAOrUIII2 小时前
【Spring Boot】控制器Controller方法
java·spring boot·后端
Dolphin_Home2 小时前
从理论到实战:图结构在仓库关联业务中的落地(小白→中级,附完整代码)
java·spring boot·后端·spring cloud·database·广度优先·图搜索算法
等....2 小时前
Miniconda使用
开发语言·python
zfj3212 小时前
go为什么设计成源码依赖,而不是二进制依赖
开发语言·后端·golang