Springboot?-?Fat?Jar示例詳解
導(dǎo)讀
Spring Boot應(yīng)用可以使用spring-boot-maven-plugin
快速打包,構(gòu)建一個可執(zhí)行jar。Spring Boot內(nèi)嵌容器,通過java -jar
命令便可以直接啟動應(yīng)用。
雖然是一個簡單的啟動命令,背后卻藏著很多知識。今天帶著大家探索FAT JAR啟動的背后原理。本文主要包含以下幾個部分:
- JAR 是什么。首先需要了解jar是什么,才知道
java -jar
做了什么事情。 - FatJar 有什么不同。 Spring Boot提供的可執(zhí)行jar與普通的jar有什么區(qū)別。
- 啟動時的類加載原理。 啟動過程中類加載器做了什么?Spring Boot又如何通過自定義類加載器解決內(nèi)嵌包的加載問題。
- 啟動的整個流程。最后整合前面三部分的內(nèi)容,解析源碼看如何完成啟動。
JAR 是什么
JAR簡介
JAR文件(Java歸檔,英語: Java ARchive)是一種軟件包文件格式,通常用于將大量的Java類文件、相關(guān)的元數(shù)據(jù)和資源(文本、圖片等)文件聚合到一個文件,以便分發(fā)Java平臺應(yīng)用軟件或庫。簡單點理解其實就是一個壓縮包,既然是壓縮包那么為了提取JAR文件的內(nèi)容,可以使用任何標準的unzip解壓縮軟件提取內(nèi)容。或者使用Java虛擬機自帶命令jar -xf foo.jar
來解壓相應(yīng)的jar文件。
JAR 可以簡單分為兩類:
- 非可執(zhí)行JAR。打包時,不用指定
main-class
,也不可運行。普通jar包可以供其它項目進行依賴。 - 可執(zhí)行JAR。打jar包時,指定了
main-class
類,可以通過java -jar xxx.jar
命令,執(zhí)行main-class
的main
方法,運行jar包。可運行jar包不可被其他項目進行依賴。
JAR結(jié)構(gòu)
包結(jié)構(gòu)
不管是非可行JAR還是可執(zhí)行JAR解壓后都包含兩部分:META-INF
目錄(元數(shù)據(jù))和package
目錄(編譯后的class)。這種普通的jar不包含第三方依賴包,只包含應(yīng)用自身的配置文件、class 等。
. ├── META-INF │ ├── MANIFEST.MF #定義 └── org # 包路徑(存放編譯后的class) └── springframework
描述文件MANIFEST.MF
JAR包的配置文件是META-INF
文件夾下的MANIFEST.MF
文件。主要配置信息如下:
- Manifest-Version: 用來定義manifest文件的版本,例如:Manifest-Version: 1.0
- Created-By: 聲明該文件的生成者,一般該屬性是由jar命令行工具生成的,例如:Created-By: Apache Ant 1.5.1
- Signature-Version: 定義jar文件的簽名版本
- Class-Path: 應(yīng)用程序或者類裝載器使用該值來構(gòu)建內(nèi)部的類搜索路徑,可執(zhí)行jar包里需要設(shè)置這個。
上面是普通jar包的屬性,可運行jar包的.MF文件中,還會有mian-class
或start-class
等屬性。如果依賴了外部jar包,還會在MF文件中配置lib路徑等信息。更多信息參見:maven為MANIFEST.MF文件添加內(nèi)容的方法
至于可運行jar包和普通jar包的目錄結(jié)構(gòu),沒有什么特別固定的模式,總之,無論是什么結(jié)構(gòu),在.MF文件中,配置好jar包的信息,即可正常使用jar包了。
FatJar有什么不同
什么是FatJar?
普通的jar只包含當前 jar的信息,不含有第三方 jar。當內(nèi)部依賴第三方j(luò)ar時,直接運行則會報錯,這時候需要將第三方j(luò)ar內(nèi)嵌到可執(zhí)行jar里。將一個jar及其依賴的三方j(luò)ar全部打到一個包中,這個包即為 FatJar。
SpringBoot FatJar解決方案
Spring Boot
為了解決內(nèi)嵌jar問題,提供了一套FatJar解決方案,分別定義了jar目錄結(jié)構(gòu)和 MANIFEST.MF
。在編譯生成可執(zhí)行 jar 的基礎(chǔ)上,使用spring-boot-maven-plugin
按Spring Boot 的可執(zhí)行包標準repackage
,得到可執(zhí)行的Spring Boot jar。根據(jù)可執(zhí)行jar類型,分為兩種:可執(zhí)行Jar和可執(zhí)行war。
spring-boot-maven-plugin打包過程
因為在新建的空的 SpringBoot 工程中并沒有任何地方顯示的引入或者編寫相關(guān)的類。實際上,對于每個新建的 SpringBoot 工程,可以在其 pom.xml 文件中看到如下插件:
<build> <plugins> <plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> </plugin> </plugins> </build>
這個是SpringBoot官方提供的用于打包FatJar的插件,org.springframework.boot.loader
下的類其實就是通過這個插件打進去的;
下面是此插件將 loader 相關(guān)類打入 FatJar 的一個執(zhí)行流程:
org.springframework.boot.maven#execute-> org.springframework.boot.maven#repackage -> org.springframework.boot.loader.tools.Repackager#repackage-> org.springframework.boot.loader.tools.Repackager#writeLoaderClasses-> org.springframework.boot.loader.tools.JarWriter#writeLoaderClasses
最終的執(zhí)行方法就是下面這個方法,通過注釋可以看出,該方法的作用就是將 spring-boot-loader 的classes 寫入到 FatJar 中。
/** * Write the required spring-boot-loader classes to the JAR. * @throws IOException if the classes cannot be written */ @Override public void writeLoaderClasses() throws IOException { writeLoaderClasses(NESTED_LOADER_JAR); }
打包結(jié)果
Spring Boot項目被編譯以后,在targert
目錄下存在兩個jar文件:一個是xxx.jar
和xxx.jar.original
。
- 其中
xxx.jar.original
是maven編譯后的原始jar文件,即標準的java jar。該文件僅包含應(yīng)用本地資源。 如果單純使用這個jar,無法正常運行,因為缺少依賴的第三方資源。 - 因此
spring-boot-maven-plugin
插件對這個xxx.jar.original
再做一層加工,引入第三方依賴的jar包等資源,將其"repackage"
為xxx.jar
。可執(zhí)行Jar的文件結(jié)構(gòu)如下圖所示:
. ├── BOOT-INF │ ├── classes │ │ ├── application.properties # 用戶-配置文件 │ │ └── com │ │ └── glmapper │ │ └── bridge │ │ └── boot │ │ └── BootStrap.class # 用戶-啟動類 │ └── lib │ ├── jakarta.annotation-api-1.3.5.jar │ ├── jul-to-slf4j-1.7.28.jar │ ├── log4j-xxx.jar # 表示 log4j 相關(guān)的依賴簡寫 ├── META-INF │ ├── MANIFEST.MF │ └── maven │ └── com.glmapper.bridge.boot │ └── guides-for-jarlaunch │ ├── pom.properties │ └── pom.xml └── org └── springframework └── boot └── loader ├── ExecutableArchiveLauncher.class ├── JarLauncher.class ├── LaunchedURLClassLoader$UseFastConnectionExceptionsEnumeration.class ├── LaunchedURLClassLoader.class ├── Launcher.class ├── MainMethodRunner.class ├── PropertiesLauncher$1.class ├── PropertiesLauncher$ArchiveEntryFilter.class ├── PropertiesLauncher$PrefixMatchingArchiveFilter.class ├── PropertiesLauncher.class ├── WarLauncher.class ├── archive │ ├── # 省略 ├── data │ ├── # 省略 ├── jar │ ├── # 省略 └── util └── SystemPropertyUtils.class
- META-INF: 存放元數(shù)據(jù)。MANIFEST.MF 是 jar 規(guī)范,Spring Boot 為了便于加載第三方 jar 對內(nèi)容做了修改;
- org: 存放Spring Boot 相關(guān)類,比如啟動時所需的 Launcher 等;
- BOOT-INF/class: 存放應(yīng)用編譯后的 class 文件;
- BOOT-INF/lib: 存放應(yīng)用依賴的 JAR 包。
Spring Boot的MANIFEST.MF
和普通jar有些不同:
Spring-Boot-Version: 2.1.3.RELEASE Main-Class: org.springframework.boot.loader.JarLauncher Start-Class: com.rock.springbootlearn.SpringbootLearnApplication Spring-Boot-Classes: BOOT-INF/classes/ Spring-Boot-Lib: BOOT-INF/lib/ Build-Jdk: 1.8.0_131
Main-Class: 是java -jar
啟動引導(dǎo)類,但這里不是項目中的類,而是Spring Boot內(nèi)部的JarLauncher。
Start-Class: 這個才是正在要執(zhí)行的應(yīng)用內(nèi)部主類
所以java -jar
啟動的時候,加載運行的是JarLauncher。Spring Boot內(nèi)部如何通過JarLauncher 加載Start-Class 執(zhí)行呢?為了更清楚加載流程,我們先介紹下java -jar
是如何完成類加載邏輯的。
啟動時的類加載原理
這里簡單說下java -jar
啟動時是如何完成記載類加載的。Java 采用了雙親委派機制,Java語言系統(tǒng)自帶有三個類加載器:
- Bootstrap CLassloder: 最頂層的加載類,主要加載核心類庫
- Extention ClassLoader: 擴展的類加載器,加載目錄
%JRE_HOME%/lib/ext
目錄下的jar包和class文件。 還可以加載-D java.ext.dirs選項指定的目錄。 - AppClassLoader: 是應(yīng)用加載器。
默認情況下通過java -classpath
,java -cp
,java -jar
使用的類加載器都是AppClassLoader。 普通可執(zhí)行jar通過java -jar
啟動后,使用AppClassLoader加載Main-class
類。 如果第三方j(luò)ar不在AppClassLoader里,會導(dǎo)致啟動時候會報ClassNotFoundException。
例如在Spring Boot可執(zhí)行jar的解壓目錄下,執(zhí)行應(yīng)用的主函數(shù),就直接報該錯誤:
Exception in thread "main" java.lang.NoClassDefFoundError: org/springframework/boot/SpringApplication
at com.glmapper.bridge.boot.BootStrap.main(BootStrap.java:13)
Caused by: java.lang.ClassNotFoundException: org.springframework.boot.SpringApplication
at java.net.URLClassLoader.findClass(URLClassLoader.java:381)
at java.lang.ClassLoader.loadClass(ClassLoader.java:424)
at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:338)
at java.lang.ClassLoader.loadClass(ClassLoader.java:357)
... 1 more
從異常堆棧來看,是因為找不到SpringApplication
這個類;這里其實還是比較好理解的,BootStrap
類中引入了SpringApplication
,但是這個類是在BOOT-INF/lib
下的,而java指令在啟動時未指明classpath
,依賴的第三方j(luò)ar無法被加載。
Spring Boot JarLauncher啟動時,會將所有依賴的內(nèi)嵌 jar (BOOT-INF/lib 目錄下) 和class(BOOT-INF/classes 目錄)都加入到自定義的類加載器LaunchedURLClassLoader中,并用這個ClassLoder去加載MANIFEST.MF配置Start-Class,則不會出現(xiàn)類找不到的錯誤。
LaunchedURLClassLoader是URLClassLoader的子類, URLClassLoader會通過URL[] 來搜索類所在的位置。Spring Boot 則將所需要的內(nèi)嵌文檔組裝成URL[],最終構(gòu)建LaunchedURLClassLoader類。
啟動的整個流程
有了以上知識的鋪墊,我們看下整個 FatJar 啟動的過程會是怎樣。為了以便查看源碼和遠程調(diào)試,可以在 pom.xml 引入下面的配置:
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-loader</artifactId> </dependency>
簡單概括起來可以分為幾步:
- java -jar 啟動,AppClassLoader 則會加載 MANIFEST.MF 配置的Main-Class, JarLauncher。
- JarLauncher啟動時,注冊URL關(guān)聯(lián)協(xié)議。
- 獲取所有內(nèi)嵌的存檔(內(nèi)嵌jar和class)
- 根據(jù)存檔的URL[]構(gòu)建類加載器。
- 然后用這個類加載器加載Start-Class。 保證這些類都在同一個ClassLoader中。
參考資料
聊一聊 SpringBoot 中 FatJar 啟動原理
Spring Boot 解析(二):FatJar 啟動原理
Springboot - Fat Jar
到此這篇關(guān)于Springboot - Fat Jar詳解的文章就介紹到這了,更多相關(guān)Springboot Fat Jar內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java代碼的三根頂梁柱:循環(huán)結(jié)構(gòu)
這篇文章主要介紹了JAVA 循環(huán)結(jié)構(gòu)的相關(guān)資料,文中講解的非常細致,示例代碼幫助大家更好的理解和學習,感興趣的朋友可以了解下2021-08-08eclipse+maven+spring mvc項目基本搭建過程
這篇文章主要介紹了eclipse+maven+spring mvc項目基本搭建過程,本文圖文并茂給大家介紹的非常詳細,具有一定的參考借鑒價值,需要的朋友可以參考下2019-09-09LocalDateTime日期時間格式中間多了一個T的問題及解決
這篇文章主要介紹了LocalDateTime日期時間格式中間多了一個T的問題及解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-03-03淺析Java中print、printf、println的區(qū)別
以下是對Java中print、printf、println的區(qū)別進行了詳細的分析介紹,需要的朋友可以過來參考下2013-08-08詳解IDEA2020新建spring項目和c3p0連接池的創(chuàng)建和使用
C3P0是一個開源的JDBC連接池,它實現(xiàn)了數(shù)據(jù)源和JNDI綁定,本文就使用Spring實現(xiàn),具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-08-08struts2如何使用攔截器進行用戶權(quán)限控制實例
本篇文章主要介紹了struts2如何使用攔截器進行用戶權(quán)限控制實例,非常具有實用價值,需要的朋友可以參考下2017-05-05