SpringBoot工程打包后執(zhí)行Java?-Jar就能啟動(dòng)的步驟原理
本文主要分享SpringBoot工程項(xiàng)目如何打包成一個(gè)可直接通過java -jar執(zhí)行的jar,并且簡(jiǎn)單分析其啟動(dòng)步驟原理。
1.SpringBoot如何打包成一個(gè)可執(zhí)行jar?
SpringBoot打包成成一個(gè)可執(zhí)行jar需要依賴一個(gè)maven打包插件spring-boot-maven-plugin,如下所示在pom文件結(jié)尾的build節(jié)點(diǎn)添加依賴,同時(shí)將src/main/java和src/main/resources打入jar里面。
<build>
<resources>
<resource>
<directory>src/main/java</directory>
<includes>
<include>**/*.*</include>
</includes>
</resource>
<resource>
<directory>src/main/resources</directory>
<includes>
<include>**/*.*</include>
</includes>
</resource>
</resources>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>此時(shí)再執(zhí)行maven package打包成jar(最后打包完成在target目錄下):

此時(shí)我們直接通過cmd控制臺(tái)執(zhí)行命令:java -jar boot-first-1.0.0-SNAPSHOT.jar 可以看到啟動(dòng)成功。

相關(guān)日志已經(jīng)打印出來了,此時(shí)我們?cè)偻ㄟ^postman驗(yàn)證一下編碼中的接口是否生效:

程序中的接口代碼如下:

2.SpringBoot打包成的jar為何可以直接Java -jar執(zhí)行?
java程序的入口是main方法 ,如果一個(gè)jar能夠通過java -jar執(zhí)行起來,肯定離不開main方法。那springboot通過spring-boot-maven-plugin插件打包成的jar通常稱為fat jar,里面不僅僅包含了程序代碼還包括其他引用的依賴的jar。那這個(gè)fat jar的main方法入口在哪兒呢?我們通過解壓該jar可以看到,在META-INF下的有一個(gè)MANIFEST.MF文件:

其中MANIFEST.MF中內(nèi)容如下,其中比較重要的幾行信息:
Start-Class: com.xren.bootfirst.BootFirstApplication ##程序開始類
Spring-Boot-Classes: BOOT-INF/classes/ ##加載的class
Spring-Boot-Lib: BOOT-INF/lib/ ##加載的內(nèi)部jar位置
Main-Class: org.springframework.boot.loader.JarLauncher ## boot啟動(dòng)類

綜上可知,SpringBoot項(xiàng)目通過引入打包插件spring-boot-maven-plugin生成特定的主清單文件MANIFEST.MF,其中包含了程序的入口類和相關(guān)啟動(dòng)依賴。
3.一窺SpringBoot初啟動(dòng)
通用的jar包如果能被java -jar執(zhí)行,只需要其MANIFEST.MF文件中有 Main-Class配置項(xiàng)即可。而此處SpringBoot的jar中的MANIFEST.MF文件里面不僅僅包含Main-Class 還有Start-Class、Spring-Boot-Classes、Spring-Boot-Lib等相關(guān)信息,那他們是如何運(yùn)行的呢?我們還是要回到該jar主類org.springframework.boot.loader.JarLauncher來一窺究竟。
查看org.springframework.boot.loader.JarLauncher類需要pom引入spring-boot-loader,如下:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-loader</artifactId>
<version>2.2.7.RELEASE</version>
</dependency>引入后其中JarLaunche類代碼如下,它包含一個(gè)main方法,并且繼承一個(gè)父類ExecutableArchiveLauncher。
public class JarLauncher extends ExecutableArchiveLauncher {
static final String BOOT_INF_CLASSES = "BOOT-INF/classes/";
static final String BOOT_INF_LIB = "BOOT-INF/lib/";
public JarLauncher() {
}
protected JarLauncher(Archive archive) {
super(archive);
}
protected boolean isNestedArchive(Entry entry) {
return entry.isDirectory() ? entry.getName().equals("BOOT-INF/classes/") : entry.getName().startsWith("BOOT-INF/lib/");
}
public static void main(String[] args) throws Exception {
(new JarLauncher()).launch(args);
}
}從main方法進(jìn)入我們需要關(guān)注一個(gè)launch方法,其中的三行
- 注冊(cè)一個(gè)jar處理類
- 獲取一個(gè)類加載器(通過路徑加載lib和classs)
- 通過獲取主啟動(dòng)類和類加載器運(yùn)行啟動(dòng)。
protected void launch(String[] args) throws Exception {
JarFile.registerUrlProtocolHandler();
ClassLoader classLoader = this.createClassLoader(this.getClassPathArchives());
this.launch(args, this.getMainClass(), classLoader);
}追尋this.getMainClass() 方法,其會(huì)指向上述的父類ExecutableArchiveLauncher中,此處就回到了我們開始說的Start-Class配置項(xiàng)。
protected String getMainClass() throws Exception {
Manifest manifest = this.archive.getManifest();
String mainClass = null;
if (manifest != null) {
mainClass = manifest.getMainAttributes().getValue("Start-Class");
}
if (mainClass == null) {
throw new IllegalStateException("No 'Start-Class' manifest entry specified in " + this);
} else {
return mainClass;
}
}getMainClass()方法中的this.archive.getManifest() 就是獲取jar下的META-INF/MANIFEST.MF文件。此處追尋到Archive的子類ExplodedArchive中,如下代碼獲取清單文件:
private File getManifestFile(File root) {
File metaInf = new File(root, "META-INF");
return new File(metaInf, "MANIFEST.MF");
}到此時(shí)清單文件中的Start-Class、Spring-Boot-Classes、Spring-Boot-Lib都已找到,準(zhǔn)備啟動(dòng)!
回到org.springframework.boot.loader.Launcher的launch方法。
- 當(dāng)前線程設(shè)置類加載器
- 創(chuàng)建主方法并運(yùn)行(主方法即為Start-Class指向的類)
protected void launch(String[] args, String mainClass, ClassLoader classLoader) throws Exception {
Thread.currentThread().setContextClassLoader(classLoader);
this.createMainMethodRunner(mainClass, args, classLoader).run();
}以上兩行代碼具體實(shí)現(xiàn)在org.springframework.boot.loader.MainMethodRunner類中,run方法即為最后的啟動(dòng)邏輯
- 通過當(dāng)前線程類加載器載入清單文件Start-Class配置的主類。
- 加載后獲取主類中的main方法(XXXApplication中的main方法)。
- 反射執(zhí)行該XXXApplication中的main方法。
public MainMethodRunner(String mainClass, String[] args) {
this.mainClassName = mainClass;
this.args = args != null ? (String[])args.clone() : null;
}
public void run() throws Exception {
Class<?> mainClass = Thread.currentThread().getContextClassLoader().loadClass(this.mainClassName);
Method mainMethod = mainClass.getDeclaredMethod("main", String[].class);
mainMethod.invoke((Object)null, this.args);
}至此,從執(zhí)行java -jar命令,SpringBoot打包的jar啟動(dòng)到程序中XXXApplication中的main方法這一階段我們就初步分析完了!
總結(jié):SpringBoot通過引入打包插件spring-boot-maven-plugin將其相關(guān)信息寫入到j(luò)ava-jar的META-INF/MANIFEST.MF文件中,而后通過清單文件中的主入口Main-Class配置的org.springframework.boot.loader.JarLauncher類加載相關(guān)class、lib和應(yīng)用程序主類Start-Class配置的XXXApplication類,再反射獲取XXXApplication類中我們業(yè)務(wù)定義的main方法啟動(dòng)運(yùn)行。
到此這篇關(guān)于SpringBoot工程打包后為何執(zhí)行Java -Jar就能啟動(dòng)?的文章就介紹到這了,更多相關(guān)springboot打包執(zhí)行ava -Jar啟動(dòng)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
springboot讀取resource配置文件生成容器對(duì)象的示例代碼
這篇文章主要介紹了springboot讀取resource配置文件生成容器對(duì)象的示例代碼,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-07-07
Mybatis使用JSONObject接收數(shù)據(jù)庫(kù)查詢的方法
這篇文章主要介紹了Mybatis使用JSONObject接收數(shù)據(jù)庫(kù)查詢,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-12-12
Spring?Boot?基于?SCRAM?認(rèn)證集成?Kafka?的過程詳解
在本篇文章中,我們將探討如何在?Spring?Boot?應(yīng)用中集成?Kafka?并使用?SCRAM?認(rèn)證機(jī)制進(jìn)行安全連接,并實(shí)現(xiàn)動(dòng)態(tài)創(chuàng)建賬號(hào)、ACL?權(quán)限、Topic,以及生產(chǎn)者和消費(fèi)者等操作,感興趣的朋友跟隨小編一起看看吧2024-08-08
SpringCloud Zuul自定義filter代碼實(shí)例
這篇文章主要介紹了SpringCloud Zuul自定義filter代碼實(shí)例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-04-04
詳解spring batch的使用和定時(shí)器Quart的使用
spring Batch是一個(gè)基于Spring的企業(yè)級(jí)批處理框架,它通過配合定時(shí)器Quartz來輕易實(shí)現(xiàn)大批量的數(shù)據(jù)讀取或插入,并且全程自動(dòng)化,無需人員管理2017-08-08
基于Spring Boot的Logback日志輪轉(zhuǎn)配置詳解
本篇文章主要介紹了基于Spring Boot的Logback日志輪轉(zhuǎn)配置詳解,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-10-10

