SpringBoot的啓動引導類真的是XXApplication嗎?

1. 引言

SpringBoot項目中的啓動類,一般都是XXApplication,例如StatsApplication,UnionApplication。每個項目的啓動類名稱都不一樣。但是它的啓動類真的是XXApplication嗎?

2. META-INF/Manifest.mf文件

jar文件實際上是class文件的zip壓縮存檔。jar並不能表達應用程序的便籤信息.

META-INF/Manifest.mf文件提供存檔的便籤信息.

Manifest.mf有 Main-Class,用來標明jar文件的入口類。

解壓jar包,查看META-INF/Manifest.mf過程如下:

重要信息如下

 

Main-Class: org.springframework.boot.loader.JarLauncher
Start-Class: com.shanyuan.StatsApplication
複製代碼

 

也就是說: org.springframework.boot.loader.JarLauncher是SpringBoot的啓動類!

下面瀏覽下JarLauncher

3. 瀏覽JarLauncher

3.1 找到JarLauncher

進入IDEA,Ctrl+N查找JarLauncher,竟然找不到!!

進入 https://search.maven.org/classic/#advancedsearch 查詢JarLauncher

在查詢結果找到spring下的項目

確定JarLauncher位於spring-boot-loader下。爲了方便查看源碼,在pom中引入

<dependency>
	<groupId>org.springframework.boot</groupId>
	<artifactId>spring-boot-loader</artifactId>
	<scope>provided</scope>
</dependency>複製代碼

3.2.JarLauncher說明

JarLauncher作爲引導類 ,當調用java -jar 命令時,將調用main方法,實際上調用的是 JarLauncher#launch方法,該方法繼承與org.springframework.boot.loader.Launcher

簡化層次關係爲:

 

JarLauncher#launch代碼如下

 

protected void launch(String[] args) throws Exception {
   JarFile.registerUrlProtocolHandler();
   ClassLoader classLoader = createClassLoader(getClassPathArchives());
   launch(args, getMainClass(), classLoader);
}複製代碼

聚句解析

1,.JarFile.registerUrlProtocolHandler();

Spring Boot生成的FAT jar,在被java -jar 引導時,其內部的jar文件無法被sun.net. www.protocol .jar.Handler處理。

所以SpringBoot實現了,org.springframework.boot.loader.jar.Handler

JarFile.registerUrlProtocolHandler(),就註冊 org.springframework.boot.loader.jar.Handler

2. ClassLoader classLoader = createClassLoader(getClassPathArchives());

創建ClassLoader。

getClassPathArchives 核心判斷是 isNestedArchive方法。

isNestedArchive被JarLauncher覆寫了。其實現如下:

 

static final String BOOT_INF_CLASSES = "BOOT-INF/classes/";

static final String BOOT_INF_LIB = "BOOT-INF/lib/";
@Override
protected boolean isNestedArchive(Archive.Entry entry) {
   if (entry.isDirectory()) {
      return entry.getName().equals(BOOT_INF_CLASSES);
   }
   return entry.getName().startsWith(BOOT_INF_LIB);
}複製代碼

也就是說,只要 滿足以BOOT-INF/classes/和BOOT-INF/lib/都是classLoader加載的範圍。

解壓的jar,查看也與只對應

3. launch(args, getMainClass(), classLoader);

 

protected void launch(String[] args, String mainClass, 
                ClassLoader classLoader)
      throws Exception {
   Thread.currentThread().setContextClassLoader(classLoader);
   createMainMethodRunner(mainClass, args, classLoader).run();
}複製代碼

查看createMainMethodRunner的run方法,如下:

public class MainMethodRunner {
    // 省略部分代碼
    public void run() throws Exception {
   Class<?> mainClass = Thread.currentThread().getContextClassLoader()
         .loadClass(this.mainClassName);
       Method mainMethod = 
           mainClass.getDeclaredMethod("main", String[].class);
       mainMethod.invoke(null, new Object[] { this.args });
    }
}複製代碼

其中mainClass,來自/META-INF/MANIFEST.MF中的Start-Class屬性。

即,JarLauncher是同進程內,通過反射調用Start-Class對應類,即XXXApplication的main方法。

4.總結

 

SpringBoot項目的實際啓動類是org.springframework.boot.loader.JarLauncher。

在JarLauncher內部通過反射調用XXApplication類的main方法。

具體實現位於 MainMethodRunner中。


 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章