欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

spring-boot-maven-plugin插件打包和java -jar命令執(zhí)行原理分析

 更新時間:2025年05月07日 09:18:36   作者:Armyyyyy丶  
這篇文章主要介紹了spring-boot-maven-plugin插件打包和java -jar命令執(zhí)行原理,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教

1. Maven生命周期

Maven的生命周期有三種:

  1. clean:清除項目構(gòu)建數(shù)據(jù),較為簡單,不深入探討;
  2. site:建立和部署項目站點,使用的較少,也不深入探討;
  3. default:定義了項目構(gòu)建時所需要的所有步驟,是Maven生命周期中最核心最重要的的部分。

本次要深入了解的便是default流程。其生命周期如下:

階段可否執(zhí)行說明
validate驗證項目是否正確以及所有必要信息是否可用
initializeX初始化構(gòu)建狀態(tài)
generate-sourcesX生成編譯階段需要的所有源碼文件
process-sourcesX處理源碼文件,例如過濾某些值
generate-resourcesX生成項目打包階段需要的資源文件
process-resourcesX處理資源文件,并復(fù)制到輸出目錄,為打包階段做準(zhǔn)備
compile編譯源代碼,并移動到輸出目錄
process-classesX處理編譯生成的字節(jié)碼文件
generate-test-sourcesX生成編譯階段需要的測試源代碼
process-test-sourcesX處理測試資源,并復(fù)制到測試輸出目錄
test-compileX編譯測試源代碼并移動到測試輸出目錄中
test使用適當(dāng)?shù)膯卧獪y試框架(如junit)運行測試
prepare-packageX在真正打包前執(zhí)行一些必要的操作
package獲取編譯后的代碼,并按照可發(fā)布的格式進(jìn)行打包,如jar、war或ear文件
pre-integration-testX在集成測試執(zhí)行之前,執(zhí)行所需的操作,例如設(shè)置環(huán)境變量
integration-testX處理和部署所需的包到集成測試能夠運行的環(huán)境中
post-integration-testX在集成測試被執(zhí)行后執(zhí)行必要的操作,例如清理環(huán)境
verify對集成測試的結(jié)果進(jìn)行檢查,以保證質(zhì)量達(dá)標(biāo)
install安裝打包的項目到本地倉庫,以供本地其它項目使用
deploy拷貝最終的包文件到遠(yuǎn)程倉庫中,以共享給其它開發(fā)人員和項目

其中可以在Maven常見的Lifecycle中直接執(zhí)行的有validate、compile、test、package、verify和deploy七種,一般在Maven的plugin標(biāo)簽中,可以通過配置如下配置來指定插件在某個階段生效,需要注意的是不可隨意配置,每個插件可處理的階段都是不同的。(不配置則執(zhí)行插件默認(rèn)的)

<executions>
    <execution>
        <phase>XX</phase>
        <goals>
            <goal>XXXX</goal>
        </goals>
    </execution>
</executions>

今天要深入了解的spring-boot-maven-plugin插件就是在package階段中生效的。

2. jar包結(jié)構(gòu)

通常而言,jar包分為可執(zhí)行jar包和不可執(zhí)行jar包,顧名思義,可執(zhí)行jar包即可通過命令java -jar直接執(zhí)行,不可執(zhí)行jar包通過命令java -jar執(zhí)行則會報錯。

2.1 不可執(zhí)jar包結(jié)構(gòu)

|-- _jar包根目錄
    |-- 原項目class文件和resource文件
    |-- _META-INF
        |-- MANIFEST.MF
        |-- _maven
            |-- _項目目錄
                |-- pom.properties
                |-- pom.xml

上面是經(jīng)典的不可執(zhí)行jar包目錄,其中MANIFEST.MF文件內(nèi)容如下:

Manifest-Version: 1.0
Archiver-Version: Plexus Archiver
Built-By: xxxxx
Created-By: Apache Maven 3.5.0
Build-Jdk: 1.8.0_151

這五項是最基本的,如果使用java -jar執(zhí)行這些jar包,將會拋出錯誤碼java.launcher.jar.error3,意為沒找到Main-Class屬性。不同語言展示的最終描述不同,由launcher+對應(yīng)語言類轉(zhuǎn)換,簡體中文在launcher_zh_CN類中轉(zhuǎn)換,{0}為jar包名稱,內(nèi)容如下:

{0}中沒有主清單屬性

英文在launcher類中轉(zhuǎn)換,{0}為jar包名稱,內(nèi)容如下:

no main manifest attribute, in {0}

2.2 可執(zhí)行jar包結(jié)構(gòu)

可執(zhí)行jar包結(jié)構(gòu)挑選經(jīng)典的springboot啟動包來做示范:

|-- _jar包根目錄
    |-- _BOOT-INF
        |-- _classes
            |-- 原項目class文件和resource文件
        |-- _lib
            |--原項目依賴的jar庫文件
    |-- _META-INF
        |-- MANIFEST.MF
        |-- spring-configuration-metadata.json(springboot項目特有)
        |-- build-info.properties
        |-- _maven
            |-- _項目目錄
                |-- pom.properties
                |-- pom.xml

上一節(jié)我們得知了如果在MANIFEST.MF中沒有Main-Class屬性,使用java -jar命令執(zhí)行jar包會報錯,接下來看看在可執(zhí)行jar包的結(jié)構(gòu),MANIFEST.MF中具體有什么屬性:

Manifest-Version: 1.0
Spring-Boot-Classpath-Index: BOOT-INF/classpath.idx
Implementation-Title: XXXX
Implementation-Version: 1.0-SNAPSHOT
Spring-Boot-Layers-Index: BOOT-INF/layers.idx
Spring-Boot-Classes: BOOT-INF/classes/
Spring-Boot-Lib: BOOT-INF/lib/
Build-Jdk-Spec: 1.8
Spring-Boot-Version: 2.1.6.RELEASE
Created-By: Maven Archiver 3.4.0
Start-Class: XXX.XXX.XXX.XXXX
Main-Class: org.springframework.boot.loader.JarLauncher

里面有兩個很重要的屬性:Start-ClassMain-Class,其中Start-Class指的是項目中springboot的SpringApplication啟動類,而Main-Class則是jar包的啟動類入口。

3. spring-boot-maven-plugin插件打包

springboot打包插件執(zhí)行原理:

  1. 讀取原jar包:Maven插件都能讀MavenProject對象內(nèi)容,從中可以讀取到Artifact信息,調(diào)用該對象的getFile()方法即可獲取原jar包文件對象;
  2. 讀取項目依賴jar庫:直接使用MavenProject對象的getArtifacts()方法即可獲取依賴的jar庫;
  3. 加載launchScript:讀取embeddedLaunchScript配置,并構(gòu)建LaunchScript對象;
  4. 重新改寫MANIFEST.MF:到此步驟開始為repackage的核心流程,改寫清單文件時最主要的便是寫入Start-ClassMain-Class屬性,除此之外還會寫入jar庫和原項目文件目錄屬性;
  5. 寫入spring-boot-loader包文件:該包是springboot對接java -jar執(zhí)行命令的核心處理邏輯,springboot打包后加入的Main-Class: org.springframework.boot.loader.JarLauncher屬性指向的類便是此包中的jar包啟動類,如果war包則會寫入war包啟動類;
  6. 寫入原項目文件:原項目文件會被挪到BOOT-INF/classes/目錄下;
  7. 寫入項目依賴jar庫:原項目依賴的jar庫會被寫入到BOOT-INF/lib/目錄下。

如果要看spring-boot-maven-plugin插件打包源碼以分析原理,可導(dǎo)入插件的依賴,此時就能看到該插件的源碼。

如果使用的是IDEA,下載源碼后打上斷點,在執(zhí)行package時,使用debug模式啟動也能直接進(jìn)行調(diào)試。

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-maven-plugin</artifactId>
    <version>XXXXX</version>
</dependency>

4. 執(zhí)行jar原理

將會分析執(zhí)行java -jar命令后,Java程序調(diào)用到Springboot啟動類main方法的流程。

  1. JVM啟動,執(zhí)行加載主函數(shù)LoadMainClass:此時是在JVM底層實現(xiàn)的,里面指定了LauncherHelper類;
  2. 執(zhí)行LauncherHelpercheckAndLoadMain方法:JVM將會調(diào)用LauncherHelpercheckAndLoadMain方法,解析并校驗jar包,并獲取主要的啟動類;
  3. 解析jar的MANIFEST.MF文件:在此方法中會完成讀取MANIFEST.MF文件,主要是讀取其中的Main-Class屬性,并做jar包啟動的校驗;
  4. GetStaticMethodID方法:JVM獲取到Main-Class類對象,調(diào)用Main-Class類對象的main方法;
  5. 執(zhí)行JarLauncher的main方法:JarLauncher繼承自Launcher,main方法最后還是會調(diào)用到Launcher.launch()方法中;
  6. 讀取jar的Start-Class:此時會讀取jar包的Start-Class屬性,該屬性就是原項目的SpringApplication啟動類;
  7. 調(diào)用啟動類的main方法:調(diào)用MainMethodRunner的run方法,里面會調(diào)用Start-Class類的main方法
  8. 此時調(diào)入到自定義的啟動類中,完成啟動Springboot程序的入口程序。

總結(jié)

以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • 深入了解Java數(shù)據(jù)結(jié)構(gòu)和算法之堆

    深入了解Java數(shù)據(jù)結(jié)構(gòu)和算法之堆

    這篇文章主要為大家介紹了Java數(shù)據(jù)結(jié)構(gòu)和算法之堆 ,具有一定的參考價值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助
    2022-01-01
  • Java實現(xiàn)控制臺輸出兩點間距離

    Java實現(xiàn)控制臺輸出兩點間距離

    這篇文章主要介紹了Java實現(xiàn)控制臺輸出兩點間距離,涉及了部分編程坐標(biāo)的問題,具有一定參考價值,需要的朋友可以了解下
    2017-09-09
  • JAVA動態(tài)維度笛卡爾積輸出的實現(xiàn)

    JAVA動態(tài)維度笛卡爾積輸出的實現(xiàn)

    本文主要介紹了JAVA動態(tài)維度笛卡爾積輸出的實現(xiàn),通過動態(tài)生成笛卡爾積,可以方便地處理多維數(shù)據(jù)集,提高數(shù)據(jù)處理效率,具有一定的參考價值,感興趣的可以了解一下
    2024-02-02
  • 解析SpringBoot中使用LoadTimeWeaving技術(shù)實現(xiàn)AOP功能

    解析SpringBoot中使用LoadTimeWeaving技術(shù)實現(xiàn)AOP功能

    這篇文章主要介紹了SpringBoot中使用LoadTimeWeaving技術(shù)實現(xiàn)AOP功能,AOP面向切面編程,通過為目標(biāo)類織入切面的方式,實現(xiàn)對目標(biāo)類功能的增強(qiáng),本文給大家介紹的非常詳細(xì),需要的朋友可以參考下
    2022-09-09
  • 基于@PathVariable注解的用法說明

    基于@PathVariable注解的用法說明

    這篇文章主要介紹了基于@PathVariable注解的用法說明,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-02-02
  • SpringBoot中的HATEOAS詳情

    SpringBoot中的HATEOAS詳情

    這篇文章主要介紹了SpringBoot中的HATEOAS詳情,SpringBoot提供了HATEOAS的便捷使用方式,文章圍繞主題展開詳細(xì)介紹內(nèi)容,需要的小伙伴可以參考一下
    2022-05-05
  • SpringBoot項目POM文件的使用小結(jié)

    SpringBoot項目POM文件的使用小結(jié)

    本文主要詳細(xì)介紹了Maven中SpringBoot項目的POM文件配置,包括項目的依賴和插件,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2024-11-11
  • 在Ubuntu系統(tǒng)下安裝JDK和Tomcat的教程

    在Ubuntu系統(tǒng)下安裝JDK和Tomcat的教程

    這篇文章主要介紹了在Ubuntu系統(tǒng)下安裝JDK和Tomcat的教程,這樣便是在Linux系統(tǒng)下搭建完整的Java和JSP開發(fā)環(huán)境,需要的朋友可以參考下
    2015-08-08
  • Java常用工具類總結(jié)

    Java常用工具類總結(jié)

    今天帶大家學(xué)習(xí)Java常用工具類,文中有非常詳細(xì)的圖文解說及代碼示例,對正在學(xué)習(xí)java的小伙伴們很有幫助,需要的朋友可以參考下
    2021-05-05
  • 深入了解Java排序算法

    深入了解Java排序算法

    本文主要介紹了深入了解Java排序算法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2007-03-03

最新評論