SpringBoot加載外部依賴過(guò)程解析
這篇文章主要介紹了SpringBoot加載外部依賴過(guò)程解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
背景
公司一個(gè)項(xiàng)目的大數(shù)據(jù)平臺(tái)進(jìn)行改造,之前使用Structured Streaming作為實(shí)時(shí)計(jì)算框架,需要替換為替換為Kafka Streams,并使用SpringBoot包裝,使其可以納入微服務(wù)體系。
然而由于之前并沒(méi)有接觸過(guò)SpringFramework相關(guān)技術(shù),并且項(xiàng)目工期較為緊張,因此只好花了2天時(shí)間看了看Spring和SpringBoot,并且在改造過(guò)程中沿用大部分原有代碼,最后套上SpringBoot的殼子(就是基本不使用Spring Data相關(guān)的封裝和DI、AOP這些特性,只是在啟動(dòng)類上添加@SpringBootApplication注解,并在main()方法里面使用SpringApplication.run()方式啟動(dòng))。
問(wèn)題
然后在部署過(guò)程中就遇到了一個(gè)比較蛋疼的問(wèn)題:因?yàn)檫z留代碼的關(guān)系,部分配置項(xiàng)的名稱是自定義的,并使用java.util.ResourceBundle的getString()來(lái)讀取,而為了讓項(xiàng)目可以注冊(cè)到Eureka以及使用SpringBoot默認(rèn)的日志配置,在配置文件中配置了eureka.client.serviceUrl.defaultZone以及l(fā)ogging.file等SpringBoot內(nèi)置的配置項(xiàng),并使用SpringBoot內(nèi)置的配置讀取方式讀取。
也就是在一個(gè)應(yīng)用程序中,混合使用了兩種讀取配置文件的方式。在使用maven將應(yīng)用程序“EatMalonPeople”打包后,根據(jù)微服務(wù)組大佬的友情提示,使用如下命令運(yùn)行程序:
java -jar EatMalonPeople.jar
然鵝運(yùn)行倒是沒(méi)問(wèn)題,不過(guò)application.properties文件在jar包里面,修改配置文件需要用vim直接修改jar包。感覺(jué)這種方式略挫,不太能接受。按照以往的經(jīng)驗(yàn),使用java -cp命令可以指定classpath,應(yīng)用程序會(huì)優(yōu)先讀取classpath指定的外部配置文件。但是當(dāng)我在config目錄下拷貝了一份application.properties文件,并修改了其中使用ResourceBundle.getString()方式讀取的配置項(xiàng)時(shí),再使用:
java -cp .:./config/application.properties EatMalonPeople.jar
啟動(dòng)后,發(fā)現(xiàn)生效的配置項(xiàng)仍然是EatMalonPeople.jar這個(gè)jar包內(nèi)的配置項(xiàng)......
感覺(jué)不太科學(xué)啊。于是去查了查SpringBoot項(xiàng)目加載配置文件的順序,結(jié)果根據(jù)Spring官網(wǎng)的提示,SpringBoot加載配置文件application.properties的順序依次為:
- 當(dāng)前目錄的cofnig目錄
- 當(dāng)前目錄
- classpath目錄下的/config目錄
- classpath目錄
但是根據(jù)這種順序,明明應(yīng)該加載config目錄下的配置文件嘛。于是在pom文件中exclude掉了配置文件:
<resources>
<resource>
<directory>src/main/resources</directory>
<excludes>
<exclude>${resource.exclude}</exclude>
</excludes>
</resource>
</resources>
這樣jar包內(nèi)就不含有配置文件了。再次使用
java -cp .:./config/application.properties EatMalonPeople.jar
啟動(dòng)后,應(yīng)用程序能注冊(cè)到Eureka,然而使用ResourceBundle.getString()方法獲取的配置項(xiàng)竟然找不到了,直接throw出了內(nèi)部封裝的找不到配置項(xiàng)錯(cuò)誤!
天啦嚕,這是要鬧哪樣嘛。原來(lái)SpringBoot讀取配置文件的順序只能保證內(nèi)部方式可以讀到,也就是說(shuō)對(duì)于SpringBoot的jar包,-cp命令是沒(méi)有用的撒。沒(méi)想到我這個(gè)剛學(xué)過(guò)兩天Spring的萌新竟然碰到了這種問(wèn)題,真是不給活路啊。
解決
又是一頓好找,終于在官網(wǎng)的另一處發(fā)現(xiàn)了原因。
原來(lái)SpringBoot是通過(guò)org.springframework.boot.loader.Launcher類來(lái)啟動(dòng)的,這貨才是jar包中META-INF/MANIFEST.MF文件中Main-Class這個(gè)屬性的值,Launcher最后會(huì)調(diào)用我們自定義啟動(dòng)類中的的main()方法(而我們自定義的啟動(dòng)類是META-INF/MANIFEST.MF文件中的Start-Class屬性的值,這個(gè)屬性應(yīng)該是SpringBoot特有的)。
這個(gè)類有三個(gè)子類,分別是JarLauncher,WarLauncher,PropertiesLauncher,前兩個(gè)Launcher都是不能添加外部依賴的。只有PropertiesLauncher是可以的。于是在spring-boot-maven-plugin中添加layout屬性,添加后的spring-boot-maven-plugin的配置是醬嬸兒的:
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<configuration>
<layout>ZIP</layout>
</configuration>
</plugin>
然后用maven重新打包,并且在運(yùn)行時(shí)使用loader.path參數(shù)指定外部classpath地址:
java -Dloader.path=./config -jar EatMalonPeople.jar
這樣SpringBoot應(yīng)用程序就可以使用兩種方式愉快的讀取外部配置文件啦(其實(shí)還有外部jar,也是可以的)!
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Spring Cloud Alibaba之Sentinel實(shí)現(xiàn)熔斷限流功能
這篇文章主要介紹了Spring Cloud Alibaba之Sentinel,這里使用阿里的sentinel來(lái)實(shí)現(xiàn)熔斷限流功能,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-04-04
java語(yǔ)言與平臺(tái)基礎(chǔ)知識(shí)點(diǎn)
在本篇文章里小編給大家整理的是一篇關(guān)于java語(yǔ)言與平臺(tái)基礎(chǔ)知識(shí)點(diǎn)內(nèi)容,有需要的朋友們跟著學(xué)習(xí)下。2019-11-11
spring?MVC實(shí)現(xiàn)簡(jiǎn)單登錄功能
這篇文章主要為大家詳細(xì)介紹了spring?MVC實(shí)現(xiàn)簡(jiǎn)單登錄功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-09-09
Java中如何將list轉(zhuǎn)為樹(shù)形結(jié)構(gòu)
這篇文章主要介紹了Java中如何將list轉(zhuǎn)為樹(shù)形結(jié)構(gòu),本文通過(guò)示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-09-09
Java 實(shí)現(xiàn)麥克風(fēng)自動(dòng)錄音
這篇文章主要介紹了Java 實(shí)現(xiàn)麥克風(fēng)自動(dòng)錄音的示例代碼,幫助大家更好的理解和使用Java,感興趣的朋友可以了解下2020-12-12
spring+apollo動(dòng)態(tài)獲取yaml格式的配置方式
這篇文章主要介紹了spring+apollo動(dòng)態(tài)獲取yaml格式的配置方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-04-04
java排查一個(gè)線上死循環(huán)cpu暴漲的過(guò)程分析
這篇文章主要介紹了java排查一個(gè)線上死循環(huán)cpu暴漲的過(guò)程分析,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-08-08
Java的帶GUI界面猜數(shù)字游戲的實(shí)現(xiàn)示例
這篇文章主要介紹了Java的帶GUI界面猜數(shù)字游戲的實(shí)現(xiàn)示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-12-12

