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