這期內(nèi)容當(dāng)中小編將會給大家?guī)碛嘘P(guān)SpringBoot FatJar的啟動原理是什么,文章內(nèi)容豐富且以專業(yè)的角度為大家分析和敘述,閱讀完這篇文章希望大家可以有所收獲。
創(chuàng)新互聯(lián)是一家集網(wǎng)站建設(shè),扶余企業(yè)網(wǎng)站建設(shè),扶余品牌網(wǎng)站建設(shè),網(wǎng)站定制,扶余網(wǎng)站建設(shè)報(bào)價(jià),網(wǎng)絡(luò)營銷,網(wǎng)絡(luò)優(yōu)化,扶余網(wǎng)站推廣為一體的創(chuàng)新建站企業(yè),幫助傳統(tǒng)企業(yè)提升企業(yè)形象加強(qiáng)企業(yè)競爭力??沙浞譂M足這一群體相比中小企業(yè)更為豐富、高端、多元的互聯(lián)網(wǎng)需求。同時(shí)我們時(shí)刻保持專業(yè)、時(shí)尚、前沿,時(shí)刻以成就客戶成長自我,堅(jiān)持不斷學(xué)習(xí)、思考、沉淀、凈化自己,讓我們?yōu)楦嗟钠髽I(yè)打造出實(shí)用型網(wǎng)站。
java中描述資源常使用URL。而URL有一個(gè)方法用于打開鏈接java.net.URL#openConnection()
。由于URL用于表達(dá)各種各樣的資源,打開資源的具體動作由java.net.URLStreamHandler
這個(gè)類的子類來完成。根據(jù)不同的協(xié)議,會有不同的handler實(shí)現(xiàn)。而JDK內(nèi)置了相當(dāng)多的handler實(shí)現(xiàn)用于應(yīng)對不同的協(xié)議。比如jar
、file
、http
等等。URL內(nèi)部有一個(gè)靜態(tài)HashTable
屬性,用于保存已經(jīng)被發(fā)現(xiàn)的協(xié)議和handler實(shí)例的映射。
獲得URLStreamHandler
有三種方法
實(shí)現(xiàn)URLStreamHandlerFactory
接口,通過方法URL.setURLStreamHandlerFactory
設(shè)置。該屬性是一個(gè)靜態(tài)屬性,且只能被設(shè)置一次。
直接提供URLStreamHandler
的子類,作為URL的構(gòu)造方法的入?yún)⒅弧5窃贘VM中有固定的規(guī)范要求:
子類的類名必須是 Handler ,同時(shí)最后一級的包名必須是協(xié)議的名稱。比如自定義了Http的協(xié)議實(shí)現(xiàn),則類名必然為xx.http.Handler
JVM 啟動的時(shí)候,需要設(shè)置 java.protocol.handler.pkgs
系統(tǒng)屬性,如果有多個(gè)實(shí)現(xiàn)類,那么中間用 | 隔開。因?yàn)镴VM在嘗試尋找Handler時(shí),會從這個(gè)屬性中獲取包名前綴,最終使用包名前綴.協(xié)議名.Handler
,使用Class.forName
方法嘗試初始化類,如果初始化成功,則會使用該類的實(shí)現(xiàn)作為協(xié)議實(shí)現(xiàn)。
SpringBoot定義了一個(gè)接口用于描述資源,也就是org.springframework.boot.loader.archive.Archive
。該接口有兩個(gè)實(shí)現(xiàn),分別是org.springframework.boot.loader.archive.ExplodedArchive
和org.springframework.boot.loader.archive.JarFileArchive
。前者用于在文件夾目錄下尋找資源,后者用于在jar包環(huán)境下尋找資源。而在SpringBoot打包的fatJar中,則是使用后者。
SpringBoot使用插件
<plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> <configuration> <mainClass>com.tccdemo.Eureka</mainClass> </configuration> </plugin>
進(jìn)行打包,打包后的文件布局如下:
BOOT-INF文件夾下放的程序編譯class和依賴的jar包
org目錄下放的是SpringBoot的啟動相關(guān)包。
來看描述文件MANIFEST.MF
的內(nèi)容
Manifest-Version: 1.0 Implementation-Title: eureka Implementation-Version: 1.0-SNAPSHOT Built-By: Administrator Implementation-Vendor-Id: com.tccdemo Spring-Boot-Version: 2.0.2.RELEASE Main-Class: org.springframework.boot.loader.JarLauncher Start-Class: com.tccdemo.Eureka Spring-Boot-Classes: BOOT-INF/classes/ Spring-Boot-Lib: BOOT-INF/lib/ Created-By: Apache Maven 3.6.1 Build-Jdk: 1.8.0_201 Implementation-URL: http://www.example.com
最為顯眼的就是程序的啟動類并不是我們項(xiàng)目的啟動類,而是SpringBoot的JarLauncher
。下面會來深究下這個(gè)類的作用。
首先來看啟動方法
public static void main(String[] args) throws Exception { new JarLauncher().launch(args); }
JarLauncher
繼承于org.springframework.boot.loader.ExecutableArchiveLauncher
。該類的無參構(gòu)造方法最主要的功能就是構(gòu)建了當(dāng)前main方法所在的FatJar的JarFileArchive
對象。下面來看launch方法。該方法主要是做了2個(gè)事情:
以FatJar為file作為入?yún)?,?gòu)造JarFileArchive對象。獲取其中所有的資源目標(biāo),取得其Url,將這些URL作為參數(shù),構(gòu)建了一個(gè)URLClassLoader。
以第一步構(gòu)建的ClassLoader加載MANIFEST.MF
文件中Start-Class
指向的業(yè)務(wù)類,并且執(zhí)行靜態(tài)方法main。進(jìn)而啟動整個(gè)程序。
通過靜態(tài)方法org.springframework.boot.loader.JarLauncher#main
就可以順利啟動整個(gè)程序。這里面的關(guān)鍵在于SpringBoot自定義的classLoader能夠識別FatJar中的資源,包括有:在指定目錄下的項(xiàng)目編譯class、在指令目錄下的項(xiàng)目依賴jar。JDK默認(rèn)用于加載應(yīng)用的AppClassLoader只能從jar的根目錄開始加載class文件,并且也不支持jar in jar這種格式。
為了實(shí)現(xiàn)這個(gè)目標(biāo),SpringBoot首先從支持jar in jar中內(nèi)容讀取做了定制,也就是支持多個(gè)!/
分隔符的url路徑。SpringBoot定制了以下兩個(gè)方面:
實(shí)現(xiàn)了一個(gè)java.net.URLStreamHandler
的子類org.springframework.boot.loader.jar.Handler
。該Handler支持識別多個(gè)!/
分隔符,并且正確的打開URLConnection
。打開的Connection是SpringBoot定制的org.springframework.boot.loader.jar.JarURLConnection
實(shí)現(xiàn)。
實(shí)現(xiàn)了一個(gè)java.net.JarURLConnection
的子類org.springframework.boot.loader.jar.JarURLConnection
。該鏈接支持多個(gè)!/
分隔符,并且自己實(shí)現(xiàn)了在這種情況下獲取InputStream的方法。而為了能夠在org.springframework.boot.loader.jar.JarURLConnection
正確獲取輸入流,SpringBoot自定義了一套讀取ZipFile的工具類和方法。這部分和ZIP壓縮算法規(guī)范緊密相連,就不深入了。
能夠讀取多個(gè)!/
的url后,事情就變得很簡單了。上文提到的ExecutableArchiveLauncher
的launch
方法會以當(dāng)前的FatJar構(gòu)建一個(gè)JarFileArchive
,并且通過該對象獲取其內(nèi)部所有的資源URL,這些URL包含項(xiàng)目編譯class和依賴jar包。在構(gòu)建這些URL的時(shí)候傳入的就是SpringBoot定制的Handler。將獲取的URL數(shù)組作為參數(shù)傳遞給自定義的ClassLoaderorg.springframework.boot.loader.LaunchedURLClassLoader
。該ClassLoader繼承自UrlClassLoader。UrlClassLoader加載class就是依靠初始參數(shù)傳入的Url數(shù)組,并且嘗試Url指向的資源中加載Class文件。有了自定義的Handler,再從Url中嘗試獲取資源就變得很容易了。
至此,SpringBoot自定義的ClassLoader就能夠加載FatJar中的依賴包的class文件了。
SpringBoot提供了一個(gè)很好的思路,但是其內(nèi)部實(shí)現(xiàn)非常復(fù)雜,特別是其自行實(shí)現(xiàn)了一個(gè)ZipFIle的解析器。但是本質(zhì)上這些背后的工作都是為了能夠讀取到FatJar內(nèi)部的Jar的class文件資源。也就是只要有辦法能夠讀取這些資源其實(shí)就可以實(shí)現(xiàn)加載Class文件了。而依靠JDK本身提供的JarFile其實(shí)就可以做到了。而讀取到所有資源后,自定義一個(gè)ClassLoader加載讀取到二進(jìn)制數(shù)據(jù)進(jìn)而定義Class對象并不是很難的項(xiàng)目實(shí)現(xiàn)。當(dāng)然,SpringBoot定制的Zip解析可以在加載類階段避免頻繁的文件解壓動作,在性能上良好一些。
上述就是小編為大家分享的SpringBoot FatJar的啟動原理是什么了,如果剛好有類似的疑惑,不妨參照上述分析進(jìn)行理解。如果想知道更多相關(guān)知識,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。
分享標(biāo)題:SpringBootFatJar的啟動原理是什么
文章網(wǎng)址:http://bm7419.com/article8/ijhoip.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供微信小程序、手機(jī)網(wǎng)站建設(shè)、動態(tài)網(wǎng)站、網(wǎng)站營銷、小程序開發(fā)、虛擬主機(jī)
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)