亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Springboot?-?Fat?Jar示例詳解

 更新時間:2023年02月01日 17:16:44   作者:光陰迷客  
這篇文章主要介紹了Springboot?-?Fat?Jar詳解,Spring?Boot內(nèi)嵌容器,通過java?-jar命令便可以直接啟動應(yīng)用,今天帶著大家探索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-classmain方法,運行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-classstart-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.jarxxx.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)

    這篇文章主要介紹了JAVA 循環(huán)結(jié)構(gòu)的相關(guān)資料,文中講解的非常細致,示例代碼幫助大家更好的理解和學習,感興趣的朋友可以了解下
    2021-08-08
  • JVM運行時數(shù)據(jù)區(qū)原理解析

    JVM運行時數(shù)據(jù)區(qū)原理解析

    這篇文章主要介紹了JVM運行時數(shù)據(jù)區(qū)原理解析,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2019-08-08
  • eclipse+maven+spring mvc項目基本搭建過程

    eclipse+maven+spring mvc項目基本搭建過程

    這篇文章主要介紹了eclipse+maven+spring mvc項目基本搭建過程,本文圖文并茂給大家介紹的非常詳細,具有一定的參考借鑒價值,需要的朋友可以參考下
    2019-09-09
  • LocalDateTime日期時間格式中間多了一個T的問題及解決

    LocalDateTime日期時間格式中間多了一個T的問題及解決

    這篇文章主要介紹了LocalDateTime日期時間格式中間多了一個T的問題及解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-03-03
  • SpringBoot整合Aop全過程

    SpringBoot整合Aop全過程

    AOP(面向切面編程)技術(shù)可以高效地解決日志記錄、事務(wù)管理、權(quán)限控制等問題,日志記錄通過自定義注解和切面類,自動記錄方法調(diào)用詳情,減少重復(fù)代碼,事務(wù)管理方面,通過AOP可以在不改變業(yè)務(wù)代碼的情況下,實現(xiàn)事務(wù)的自動開啟、提交和回滾,保證數(shù)據(jù)一致性
    2024-10-10
  • JAVA中的引用與對象詳解

    JAVA中的引用與對象詳解

    本文主要介紹了JAVA中引用與對象的相關(guān)知識。具有很好的參考價值,下面跟著小編一起來看下吧
    2017-02-02
  • 淺析Java中print、printf、println的區(qū)別

    淺析Java中print、printf、println的區(qū)別

    以下是對Java中print、printf、println的區(qū)別進行了詳細的分析介紹,需要的朋友可以過來參考下
    2013-08-08
  • Spring MVC的web.xml配置詳解

    Spring MVC的web.xml配置詳解

    這篇文章主要介紹了Spring MVC的web.xml配置詳解,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-06-06
  • 詳解IDEA2020新建spring項目和c3p0連接池的創(chuàng)建和使用

    詳解IDEA2020新建spring項目和c3p0連接池的創(chuàng)建和使用

    C3P0是一個開源的JDBC連接池,它實現(xiàn)了數(shù)據(jù)源和JNDI綁定,本文就使用Spring實現(xiàn),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-08-08
  • struts2如何使用攔截器進行用戶權(quán)限控制實例

    struts2如何使用攔截器進行用戶權(quán)限控制實例

    本篇文章主要介紹了struts2如何使用攔截器進行用戶權(quán)限控制實例,非常具有實用價值,需要的朋友可以參考下
    2017-05-05

最新評論