三分鐘帶你了解SpringBoot真正的啟動引導類
引言
SpringBoot
項目中的啟動類,一般都是XXApplication
,例如**「StatsApplication」,「UnionApplication」**。
每個項目的啟動類名稱都不一樣。但是它的啟動類真的是XXApplication嗎?
**META-INF/**Manifest.mf文件
jar文件實際上是class文件的zip壓縮存檔。jar并不能表達應用程序的便簽信息.
「META-INF/Manifest.mf文件提供存檔的便簽信息.」
Manifest.mf有 「Main-Class,用來標明jar文件的入口類?!?/strong>
解壓jar包,查看META-INF/Manifest.mf過程如下:
重要信息如下
Main-Class: org.springframework.boot.loader.JarLauncher
Start-Class: com.shanyuan.StatsApplication
也就是說:「org.springframework.boot.loader.JarLauncher 是 SpringBoot 的啟動類!」
下面瀏覽下 JarLauncher
瀏覽JarLauncher
3.1 找到JarLauncher
進入IDEA,Ctrl+N查找JarLauncher,竟然找不到??!
進入 https://search.maven.org/classic/#advancedsearch 查詢JarLauncher
在查詢結(jié)果找到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
作為引導類 ,當調(diào)用java -jar
命令時,將調(diào)用 main 方法,實際上調(diào)用的是 **「JarLauncher#launch」**方法,該方法繼承于 org.springframework.boot.loader.Launcher
簡化層次關(guān)系為:
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 引導時,其內(nèi)部的 jar 文件無法被 sun.net.www.protocol.jar.Handler
處理。微信公眾號搜索, [Java學習之道] ,回復 ‘福利' 2T 資料等你來拿~
所以 SpringBoot 實現(xiàn)了,org.springframework.boot.loader.jar.Handler
JarFile.registerUrlProtocolHandler()
, 就注冊 org.springframework.boot.loader.jar.Handler
「2.ClassLoader classLoader = createClassLoader(getClassPathArchives());」
創(chuàng)建ClassLoader。
getClassPathArchives
核心判斷是 isNestedArchive
方法。
isNestedArchive 被 JarLauncher 覆寫了。其實現(xiàn)如下:
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 是同進程內(nèi),通過反射調(diào)用 Start-Class 對應類,即 XXXApplication 的 main 方法?!?/strong>
4.總結(jié)
SpringBoot 項目的實際啟動類是 org.springframework.boot.loader.JarLauncher
。
「在 JarLauncher 內(nèi)部通過反射調(diào)用 XXApplication 類的 main 方法。具體實現(xiàn)位于 MainMethodRunner中?!?/strong>
到此這篇關(guān)于三分鐘帶你了解SpringBoot真正的啟動引導類的文章就介紹到這了,更多相關(guān)SpringBoot 啟動引導類內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
解析java.library.path和LD_LIBRARY_PATH的介紹與區(qū)別
這篇文章主要介紹了java.library.path和LD_LIBRARY_PATH的介紹與區(qū)別,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-05-05java利用mybatis攔截器統(tǒng)計sql執(zhí)行時間示例
這篇文章主要介紹了java利用mybatis攔截器統(tǒng)計sql執(zhí)行時間示例,該攔截器攔截mybatis的query和update操作,能統(tǒng)計sql執(zhí)行時間2014-03-03Java中String字符串常量池和intern方法源碼分析
在之前的文章中,小編給大家介紹了String字符串的不可變性及其實現(xiàn)原理,其中給大家提到了字符串常量池的概念,那么什么是常量池,String字符串與常量池有什么關(guān)系,本文給大家嘮嘮字符串常量池及String#intern()方法的作用,需要的朋友可以參考下2023-05-05