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模块镜像中的资源了

相关推荐
好家伙VCC4 分钟前
# 发散创新:用 Rust构建高性能游戏日系统,从零实现事件驱动架构 在现代游戏开发中,**性能与可扩展性**是核心命题。传统基于
java·python·游戏·架构·rust
该怎么办呢10 分钟前
Source/Core/DeveloperError.js
开发语言·javascript·ecmascript
爱笑的源码基地11 分钟前
门诊his系统源码,中西医结合的数字化门诊解决方案
java·spring boot·源码·二次开发·门诊系统·云诊所系统·诊所软件源码
庞轩px12 分钟前
缓存Key设计的“七要七不要”
java·jvm·redis·缓存
小璐资源网12 分钟前
Java 21 新特性实战:虚拟线程详解
java·开发语言·python
m0_5698814722 分钟前
基于C++的数据库连接池
开发语言·c++·算法
SimonKing24 分钟前
全网爆火的OpenClaw保姆级教程Linux版,它来了。
java·后端·程序员
.select.29 分钟前
c++ auto
开发语言·c++·算法
于慨31 分钟前
tauri
java·服务器·前端
WZTTMoon32 分钟前
从互斥锁到无锁,Java 20年并发安全进化史
java·python·安全