Spring?Boot?打包成Jar包運行原理分析
SpringBoot打包
相比與傳統(tǒng)的Java打包方式,使用SpringBoot打包插件打包成jar包后,可以直接使用java -jar
運行SpringBoot項目,本篇就來分析一下運行的原理。
SpringBoot打包插件
<plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> </plugin>
打包完后會生成兩個文件,一個***.jar
和***.jar.original
.jar
文件是SpringBoot打包后生成的文件,.jar.original
是用原生方式打包生成的文件,對比一下兩個的區(qū)別
.jar.original文件
.jar文件
.jar.original
就是普通的jar打包的結(jié)構(gòu),這里主要看.jar
文件的結(jié)構(gòu):
META-INFO目錄:META-INFO/MANIFEST.MF里包含了jar包的元數(shù)據(jù),包含了項目的啟動類等信息.
org目錄:該目錄下包含的是啟動項目的一些類,啟動的過程就在這個包里。
BOOT-INFO目錄:本地項目的代碼(BOOT-INF/classes),以及所需的以依賴(BOOT-INFO/lib)
重點 META-INFO/MANIFEST.MF
Manifest-Version: 1.0 Spring-Boot-Classpath-Index: BOOT-INF/classpath.idx Implementation-Title: demo Implementation-Version: 0.0.1-SNAPSHOT Spring-Boot-Layers-Index: BOOT-INF/layers.idx Start-Class: com.example.demo.DemoApplication Spring-Boot-Classes: BOOT-INF/classes/ Spring-Boot-Lib: BOOT-INF/lib/ Build-Jdk-Spec: 1.8 Spring-Boot-Version: 2.4.5 Created-By: Maven Jar Plugin 3.2.0 Main-Class: org.springframework.boot.loader.JarLauncher
這里有幾個重點的字段
- Main-Class :jar包啟動類,這是java規(guī)定的字段,存在這個字段的情況下, 在
java -jar
時,jar包才會運行起來 - Start-Class:本地項目的啟動類
- Spring-Boot-Classes:加載應(yīng)用類的入口
- Spring-Boot-Lib:項目所需的依賴
有了Main-Class啟動類,那就直接進(jìn)入到JarLauncher
里查看運行的過程
public static void main(String[] args) throws Exception { (new JarLauncher()).launch(args); }
在JarLauncher
的main
方法里調(diào)用了launch
方法,launch
方法的具體實現(xiàn)在JarLauncher
的抽象父類Launcher
中實現(xiàn)
protected void launch(String[] args) throws Exception { if (!this.isExploded()) { JarFile.registerUrlProtocolHandler(); } ClassLoader classLoader = this.createClassLoader(this.getClassPathArchivesIterator()); String jarMode = System.getProperty("jarmode"); String launchClass = jarMode != null && !jarMode.isEmpty() ? "org.springframework.boot.loader.jarmode.JarModeLauncher" : this.getMainClass(); this.launch(args, launchClass, classLoader); }
首先獲取了類加載器。
然后獲取jarMode,再根據(jù)jarMode獲取launchClass,如果沒有設(shè)置jarMode,則根據(jù)getMainClass
方法獲取,getMainClass
的具體實現(xiàn)在ExecutableArchiveLauncher
中實現(xiàn)
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
里獲取了MANIFEST.MF
文件里Start-Class
字段的值,也就是本地項目的啟動類。
最后調(diào)用this.launch(args, launchClass, classLoader);
protected void launch(String[] args, String launchClass, ClassLoader classLoader) throws Exception { Thread.currentThread().setContextClassLoader(classLoader); this.createMainMethodRunner(launchClass, args, classLoader).run(); }
調(diào)用MainMethodRunner
的run()
方法
public void run() throws Exception { Class<?> mainClass = Class.forName(this.mainClassName, false, Thread.currentThread().getContextClassLoader()); Method mainMethod = mainClass.getDeclaredMethod("main", String[].class); mainMethod.setAccessible(true); mainMethod.invoke((Object)null, this.args); }
在run()
方法里通過反射拿到了項目的啟動類的main
方法,從而啟動本地項目。
以上就是Spring Boot 打包成Jar包運行的原理的詳細(xì)內(nèi)容,更多關(guān)于Spring Boot Jar包的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Maven腳手架如何基于jeecg實現(xiàn)快速開發(fā)
這篇文章主要介紹了Maven腳手架如何基于jeecg實現(xiàn)快速開發(fā),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-10-10SpringBoot集成itextpdf實現(xiàn)根據(jù)模板動態(tài)生成PDF
這篇文章主要為大家詳細(xì)介紹了SpringBoot如何集成itextpdf實現(xiàn)根據(jù)模板動態(tài)生成PDF,文中的示例代碼講解詳細(xì),需要的小伙伴可以參考一下2024-03-03mybatis調(diào)用mysql存儲過程并獲取返回值方式
這篇文章主要介紹了mybatis調(diào)用mysql存儲過程并獲取返回值方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-08-08spring?aop?Pointcut?execution規(guī)則介紹
這篇文章主要介紹了spring?aop?Pointcut?execution規(guī)則,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-11-11