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 小时前
SimpleDateFormat既然不推荐使用,为什么java 8+中不删除此类
java
间彧1 小时前
DateTimeFormatter相比SimpleDateFormat在性能上有何差异?
java
间彧1 小时前
为什么说SimpleDateFormat是经典的线程不安全类
java
MacroZheng1 小时前
横空出世!MyBatis-Plus 同款 ES ORM 框架,用起来够优雅!
java·后端·elasticsearch
用户0332126663672 小时前
Java 查找并替换 Excel 中的数据:详细教程
java
间彧2 小时前
ThreadLocal实现原理与应用实践
java
若水不如远方2 小时前
Netty的四种零拷贝机制:深入原理与实战指南
java·netty
用户7493636848432 小时前
【开箱即用】一分钟使用java对接海外大模型gpt等对话模型,实现打字机效果
java
SimonKing3 小时前
一键开启!Spring Boot 的这些「魔法开关」@Enable*,你用对了吗?
java·后端·程序员