面試官:為什么SpringBoot的 jar 可以直接運(yùn)行?
哈嘍,大家好,我是了不起。
現(xiàn)在Java Web 開(kāi)發(fā)應(yīng)該都是使用的 SpringBoot,部署的時(shí)候直接打包成jar包運(yùn)行即可。
但是之前用SSH或SSM開(kāi)發(fā)的時(shí)候,通常是打包成war包,然后部署到類似Tomcat的web服務(wù)器運(yùn)行。
那么問(wèn)題來(lái)了:為什么 SpringBoot 的 jar 包可以直接運(yùn)行呢?
1、Fat jar
和普通的 jar 包結(jié)構(gòu)不同,通過(guò) SpringBoot 打包而成的 jar 包是 Fat jar(胖 JAR),在 BOOT-INF/lib 目錄下,包含了項(xiàng)目依賴的全部jar 包。在 BOOT-INF/classes 目錄下,包含了項(xiàng)目運(yùn)行的class文件。
這意味著這個(gè) JAR 文件不僅包含了應(yīng)用的代碼,還包括了所有必要的依賴庫(kù)。這樣做的好處就是,我們不需要在運(yùn)行環(huán)境中單獨(dú)安裝這些庫(kù)了,因?yàn)樗鼈円呀?jīng)包含在 JAR 文件中了。
圖片
可能會(huì)有人問(wèn),是如何打包成這種方式的呢?
通常在我們的構(gòu)建文件中,比如pom.xml 文件:
<project>
<!-- ... 其他配置 ... -->
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>
</project>當(dāng)我們執(zhí)行 mvn package 命令時(shí),它會(huì)創(chuàng)建一個(gè)包含所有依賴的 JAR 文件。
2、內(nèi)嵌服務(wù)器
Spring Boot 應(yīng)用通常內(nèi)嵌了一個(gè)服務(wù)器(如 Tomcat)。這是通過(guò)在 pom.xml 或 build.gradle 文件中添加相應(yīng)的依賴實(shí)現(xiàn)的。
例如,以下是一個(gè)包含 Spring Boot 與 Tomcat 依賴的 pom.xml 片段:
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<!-- ... 其他依賴 ... -->
</dependencies>當(dāng) Spring Boot 應(yīng)用啟動(dòng)時(shí),它會(huì)自動(dòng)配置并啟動(dòng)這個(gè)內(nèi)嵌的 Tomcat 服務(wù)器。
3、Spring Boot 啟動(dòng)器
每個(gè) Spring Boot 應(yīng)用都有一個(gè)入口類,這個(gè)類包含了 main 方法,它是整個(gè)應(yīng)用啟動(dòng)的起點(diǎn)。這個(gè)類使用 @SpringBootApplication 注解,這個(gè)注解是一個(gè)方便的注解,集成了 @Configuration、@EnableAutoConfiguration 和 @ComponentScan。例如:
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
@SpringBootApplication
public class MyApplication {
public static void main(String[] args) {
SpringApplication.run(MyApplication.class, args);
}
}當(dāng)運(yùn)行這個(gè)應(yīng)用時(shí),SpringApplication.run() 方法會(huì)啟動(dòng) Spring 應(yīng)用上下文,并且啟動(dòng)內(nèi)嵌的服務(wù)器。
4、可執(zhí)行 JAR 文件
在構(gòu)建的 JAR 文件中,META-INF/MANIFEST.MF 文件指定了主類(Main-Class)。
當(dāng)我們使用 java -jar 命令運(yùn)行 JAR 文件時(shí),Java 虛擬機(jī)就知道從哪個(gè)類的 main 方法開(kāi)始執(zhí)行。
這里的Main-Class是Spring Boot的JarLauncher,它是負(fù)責(zé)啟動(dòng)整個(gè)Spring Boot應(yīng)用程序的類。
大家可以追蹤一下源碼:
protected void launch(String[] args) throws Exception {
JarFile.registerUrlProtocolHandler();
//自定義類加載器加載jar文件
ClassLoader classLoader = createClassLoader(getClassPathArchives());
//關(guān)注getMainClass方法
launch(args, getMainClass(), classLoader);
}@Override
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);
}
return mainClass;
}
































