浅析 SpringBoot FatJar 机制的设计与实现

SpringBoot FatJar 的设计,打破了标准 jar 的结构,在 jar 包内携带了其所依赖的 jar 包,通过在标准 jar 包中指定的 Main-Class 的 main 方法启动后,创建自己的类加载器,来识别、加载、运行其非规范的目录下的代码(BOOT-INF/classes/…)和依赖(BOOT-INF/lib/…)。BOOT-INF/classes/ 目录下有 SpringBoot 上下文的启动类的 class 文件,自定义类加载器加载这个启动类后,开始进入 SpringBoot 的上下文中运行我们所写的程序代码。执行的流程可概括为:,打开 Java 的 jar 文件我们经常可以看到文件中包含着一个META-INF目录,这个目录下会有一些文件,其中必有一个MANIFEST.MF,这个文件描述了该 Jar 文件的很多信息 其中 Main-Class 定义 Jar 文件的入口类,该类必须是一个可执行的类,一旦定义了该属性即可通过  java -jar xxx.jar来运行该 jar 文件。,在生产环境中,是使用java -jar xxx.jar 的方式来运行 SpringBoot 程序。这种情况下,SpringBoot 应用真实的启动类并不是我们所定义的带有 main 方法的启动类,而是其内置的 JarLauncher 类。查看 SpringBoot 所打成的 fat jar,其 Main-Class 是org.springframework.boot.loader.JarLauncher,这便是微妙之处。,JAR 包中的 MANIFEST.MF 文件详解以及编写规范[1],org.springframework.boot.loader.JarLauncher这个类是哪里来的呢?答案在 spring-boot-loader-***.jar 包中,可找到这个 JarLauncher 类的源码。在项目中加入 maven 依赖,以便查看源码和远程调试。,图片图片,认真比较可以看出,这个 spring-boot-loader 包中的内容与 SpringBoot 的 FatJar 包中的一部分内容几乎一样。JarLauncher 在 jar 中的位置如下:,图片图片,重点重点重点:因 jar 规范要求 Main-Class 所指定的类必须位于 jar 包的顶层目录下,即 org.springframework.boot.loader.JarLauncher 这个 org 必须位于 jar 包中的第一级目录,不能放置在其他的目录下。所以 所以 所以(重点)只能将 spring-boot-loader 这个 jar 包的内容拷贝出来,而不是整个 jar 直接放置于执行 Jar 中。,图片图片,上边 JarLauncher 的这个 org.springframework.xx 以及 META-INF 这两个目录是符合 jar 包规范的。但是 BOOT-INF 这个目录里边有点像我们开发中的一些用法:,所以classes 和 lib 你也能意识到,这个设计是独特的。早期 jar 包内携带依赖是采用如 maven-shade-plugin 的做法,把依赖的 class 文件拷贝到目标 jar 中,但也会造成重名(全限定名)的类会出现覆盖的情况。后来 SpringBoot 为了避免提取覆盖的情况,修改了打包机制,放弃了maven-shade-plugin那种拷贝 class 的方式,调整为依赖原始 jar 包;这同时意味着改变了 jar 标准的运行机制,那么要想让classes和lib中代码能够正常运行,你试想一下如果没有自定义的类加载器来加载这些类文件,可以嘛?答案是:不可以,需要自定义类加载器。,通常自定义类加载器完成资源加载的核心逻辑为:,所以 SpringBoot FatJar 中自定义 ClassLoader 中的核心逻辑也是如此。,构造方法中基于 jar 包的文件系统信息,构造 Archive 对象,采集 jar 包中的 classes和lib目录下的归档文件。后边创建 ClassLoader 的时候作为参数传入,本文转载自微信公众号「架构染色」,可以通过以下二维码关注。转载本文请联系【架构染色】公众号作者。,浅析 SpringBoot FatJar 机制的设计与实现,

文章版权声明

 1 原创文章作者:cmcc,如若转载,请注明出处: https://www.52hwl.com/28893.html

 2 温馨提示:软件侵权请联系469472785#qq.com(三天内删除相关链接)资源失效请留言反馈

 3 下载提示:如遇蓝奏云无法访问,请修改lanzous(把s修改成x)

 免责声明:本站为个人博客,所有软件信息均来自网络 修改版软件,加群广告提示为修改者自留,非本站信息,注意鉴别

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023年6月23日
下一篇 2023年7月15日