一、概述SpringBoot FatJar 的设计,打破了标准 jar 的结构,在 jar 包内携带了其所依赖的 jar 包,通过在标准 jar 包中指定的 MAIn-Class 的 main 方法启动后,创建自己的类加载器,来识别、加载、运行其非规范的目录下的代码(BOOT-INF/classes/...)和依赖(BOOT-INF/lib/...) 。BOOT-INF/classes/ 目录下有 SpringBoot 上下文的启动类的 class 文件,自定义类加载器加载这个启动类后,开始进入 SpringBoot 的上下文中运行我们所写的程序代码 。执行的流程可概括为:
- 通过 JAVA -jar xxx.jar 启动应用
- 执行 xxx.jar 中 META-INF/MANIFEST.MF 里 Main-Class 所指定的 JarLauncher 类的 main 方法
- main 方法中创建自定义的 ClassLoader 即 LaunchedURLClassLoader,并将其设置为线程上下文类加载器
- 由 LaunchedURLClassLoader 加载 META-INF/MANIFEST.MF 里 Start-Class 所指定的 SpringBoot 应用的启动类(在 BOOT-INF/classes/目录下),调用其 main 方法,开始执行 SpringApplication.run(...)
在生产环境中,是使用java -jar xxx.jar 的方式来运行 SpringBoot 程序 。这种情况下,SpringBoot 应用真实的启动类并不是我们所定义的带有 main 方法的启动类,而是其内置的 JarLauncher 类 。查看 SpringBoot 所打成的 fat jar,其 Main-Class 是org.springframework.boot.loader.JarLauncher,这便是微妙之处 。
Spring-Boot-Version: 2.1.3.RELEASEMain-Class: org.springframework.boot.loader.JarLauncherStart-Class: com.rock.springbootlearn.SpringbootLearnApplicationSpring-Boot-Classes: BOOT-INF/classes/Spring-Boot-Lib: BOOT-INF/lib/Build-Jdk: 1.8.0_131JAR 包中的 MANIFEST.MF 文件详解以及编写规范[1]三、探索 JarLauncherorg.springframework.boot.loader.JarLauncher这个类是哪里来的呢?答案在 spring-boot-loader-***.jar 包中,可找到这个 JarLauncher 类的源码 。在项目中加入 maven 依赖,以便查看源码和远程调试 。
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-loader</artifactId></dependency>【浅析 SpringBoot FatJar 机制的设计与实现】
文章插图
图片
认真比较可以看出,这个 spring-boot-loader 包中的内容与 SpringBoot 的 FatJar 包中的一部分内容几乎一样 。JarLauncher 在 jar 中的位置如下:

文章插图
图片
3.1 只能拷贝出来一份儿重点重点重点:因 jar 规范要求 Main-Class 所指定的类必须位于 jar 包的顶层目录下,即 org.springframework.boot.loader.JarLauncher 这个 org 必须位于 jar 包中的第一级目录,不能放置在其他的目录下 。所以 所以 所以(重点)只能将 spring-boot-loader 这个 jar 包的内容拷贝出来,而不是整个 jar 直接放置于执行 Jar 中 。
3.2 携带程序所依赖的 jar 而非仅 class

文章插图
图片
上边 JarLauncher 的这个 org.springframework.xx 以及 META-INF 这两个目录是符合 jar 包规范的 。但是 BOOT-INF 这个目录里边有点像我们开发中的一些用法:
- 依赖的 jar 包在 lib 目录下
- 但按照 jar 包规范 jar 中不能有 jar 包的情况下
- 程序.class 文件在 classes 目录下
- 但 xxx.class 文件应该按照 org.springframework.xx 这样放置在 jar 中的根目录中
推荐阅读
- 聊聊 Springboot 启动原理
- HTTP请求:Requests的进阶使用方法浅析
- SpringBoot整合Mybatis-Plus多数据源
- 浅析NAS不会被网盘取代的原因
- 浅析 Redis 中 String 数据类型及其底层编码
- SpringBoot自动装配原理
- Spring/SpringBoot中的声明式事务和编程式事务源码、区别、优缺点、适用场景、实战
- SpringBoot+Vue+ES 实现仿百度全文搜索
- SpringBoot整合RocketMQ,老鸟们都是这么玩的!
- 在 SpringBoot 中使用 Spring AOP 实现接口鉴权
