避免多個jar通過maven打包導(dǎo)致同名配置文件覆蓋沖突問題
前言
不知道大家在開發(fā)的過程中,有沒有遇到這種場景,外部的項目想訪問內(nèi)部nexus私倉的jar,因為私倉不對外開放,導(dǎo)致外部的項目沒法下載到私倉的jar,導(dǎo)致項目因缺少jar而無法運行。
通常遇到這種場景,常用的解法有,外部項目跟內(nèi)部nexus的網(wǎng)絡(luò)打通,比如通過VPN。或者將私倉的jar直接下載下來給到外部項目。對于第二種方案有時候因為私倉的jar里面有依賴其他的內(nèi)部jar,導(dǎo)致要下載多個jar的情況。這時候為了方便,我們可能會將這些jar合并成一個大jar,再給出去。而目前有些jar都是一些starter,會有一些同名的配置文件,比如spring.factories。如果不進行處理,直接打包,就會出現(xiàn)同名配置文件覆蓋的情況
本文就是要來聊聊當多個jar合并成一個jar,如何解決多個同名配置文件覆蓋的情況
解決思路
通過maven-shade-plugin這個插件,利用插件的org.apache.maven.plugins.shade.resource.AppendingTransformer來處理處理多個jar包中存在重名的配置文件的合并。他的核心是在于合并多個同名配置文件內(nèi)容,而非覆蓋
示例配置如下
<build>
<plugins>
<!-- 防止同名配置文件,在打包時被覆蓋,用來處理多個jar包中存在重名的配置文件的合并
參考dubbo:https://github.com/apache/dubbo/blob/master/dubbo-all/pom.xml-->
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-shade-plugin</artifactId>
<executions>
<execution>
<phase>package</phase>
<goals>
<goal>shade</goal>
</goals>
<configuration>
<transformers>
<transformer implementation="org.apache.maven.plugins.shade.resource.AppendingTransformer">
<resource>META-INF/spring.factories</resource>
</transformer>
<transformer implementation="org.apache.maven.plugins.shade.resource.AppendingTransformer">
<resource>META-INF/spring.handlers</resource>
</transformer>
<transformer implementation="org.apache.maven.plugins.shade.resource.AppendingTransformer">
<resource>META-INF/spring.schemas</resource>
</transformer>
<transformer implementation="org.apache.maven.plugins.shade.resource.AppendingTransformer">
<resource>META-INF/spring.tooling</resource>
</transformer>
</transformers>
</configuration>
</execution>
</executions>
</plugin>打包后的配置文件的效果如下圖

眼尖的朋友應(yīng)該發(fā)現(xiàn)了,同名的配置內(nèi)容是通過追加的方式,但僅僅追加,其實有時候還滿足不了要求,比如spring.factories文件,他需要達到的效果應(yīng)該是如下圖

后面我通過maven-shade-plugin的官方示例(https://maven.apache.org/plugins/maven-shade-plugin/examples/resource-transformers.html)試圖想找到解決方案,但是有點遺憾,沒找到。
于是在我面前就有兩條路,一條是放棄maven-shade-plugin插件,比如選擇其他類似的插件,比如maven-assembly-plugin,這種方案我試過,發(fā)現(xiàn)maven-assembly-plugin這個插件的擴展配置,比maven-shade-plugin復(fù)雜一些,于是放棄。最后選擇了在maven-shade-plugin基礎(chǔ)再擴展一下。
擴展的思路
我并沒采用直接修改maven-shade-plugin插件的方式,而是在maven-shade-plugin打包后的基礎(chǔ)上,再進行插件定制。實現(xiàn)的思路也不難,就是修改maven-shade-plugin打成jar后的spring.factories文件內(nèi)容,將

調(diào)整成形如下即可

自定義maven插件spring-factories-merge-plugin
核心思路
1、如何讀取配置文件spring.factories中key重復(fù)的內(nèi)容,而不被覆蓋
如果是直接使java.util.properties的讀取,當配置文件中有key重復(fù)時,比如有多個org.springframework.boot.autoconfigure.EnableAutoConfiguration時,最后會出現(xiàn)value值被覆蓋的情況。
解決方案,我們可以利用org.apacche.commons.configuration.PropertiesConfiguration來進行處理
在項目的pom引入GAV
<dependency>
<groupId>org.apache.commons</groupId>
<artifactId>commons-configuration2</artifactId>
<version>${commons-configuration2}</version>
</dependency>讀取配置示例代碼
@SneakyThrows
public static Map<String, Set<String>> readFactoriesFile(InputStream input) {
// 讀取 spring.factories 內(nèi)容
//利用PropertiesConfiguration取配置文件中key重復(fù)的內(nèi)容,而不被覆蓋
PropertiesConfiguration properties = new PropertiesConfiguration();
properties.read(new InputStreamReader(input));
Map<String, Set<String>> multiSetMap = new LinkedHashMap<>();
Iterator<String> keys = properties.getKeys();
while(keys.hasNext()) {
String key = keys.next();
String[] values = properties.getStringArray(key);
Set<String> collectSet = new LinkedHashSet<>();
buildKeyValues(values, collectSet);
multiSetMap.put(key,collectSet);
}
return multiSetMap;
}2、如何將修改后的配置文件,重新寫入jar
我這邊的思路就是直接利用IO進行操作了
示例如下
public static void writeFactoriesFile(String factoriesBaseClassPathDir,String finalJarName) throws IOException {
String jarFilePath = String.format(factoriesBaseClassPathDir + "/target/" + finalJarName).replace("\\", "/").replaceAll("http://+", "/");
if(!jarFilePath.endsWith(".jar")){
jarFilePath = jarFilePath + ".jar";
}
JarFile jarFile = new JarFile(jarFilePath);
if(jarFile != null){
List<JarEntry> jarFiles = jarFile.stream().collect(Collectors.toList());
@ Cleanup FileOutputStream fos = new FileOutputStream(jarFile.getName(), true);
@ Cleanup JarOutputStream jos = new JarOutputStream(fos);
for (JarEntry jarEntry : jarFiles) {
if(jarEntry.getName().startsWith(SpringFactoriesLoader.FACTORIES_RESOURCE_LOCATION)){
try {
@ Cleanup InputStream input = jarFile.getInputStream(jarEntry);
Map<String, Set<String>> factoriesMap = readFactoriesFile(input);
jos.putNextEntry(new JarEntry(jarEntry.getName()));
generateFactoriesContent(factoriesMap,jos);
} catch (IOException e) {
e.printStackTrace();
}
}else{
//表示將該JarEntry寫入jar文件中 也就是創(chuàng)建該文件夾和文件
jos.putNextEntry(new JarEntry(jarEntry));
jos.write(streamToByte(jarFile.getInputStream(jarEntry)));
}
}
}
}項目中如何配置插件
<build>
<plugins>
<!-- 防止同名配置文件,在打包時被覆蓋,用來處理多個jar包中存在重名的配置文件的合并
參考dubbo:https://github.com/apache/dubbo/blob/master/dubbo-all/pom.xml-->
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-shade-plugin</artifactId>
<executions>
<execution>
<phase>package</phase>
<goals>
<goal>shade</goal>
</goals>
<configuration>
<transformers>
<transformer implementation="org.apache.maven.plugins.shade.resource.AppendingTransformer">
<resource>META-INF/spring.factories</resource>
</transformer>
<transformer implementation="org.apache.maven.plugins.shade.resource.AppendingTransformer">
<resource>META-INF/spring.handlers</resource>
</transformer>
<transformer implementation="org.apache.maven.plugins.shade.resource.AppendingTransformer">
<resource>META-INF/spring.schemas</resource>
</transformer>
<transformer implementation="org.apache.maven.plugins.shade.resource.AppendingTransformer">
<resource>META-INF/spring.tooling</resource>
</transformer>
</transformers>
</configuration>
</execution>
</executions>
</plugin>
<plugin>
<groupId>com.github.lybgeek.jar</groupId>
<artifactId>spring-factories-merge-plugin</artifactId>
<version>0.0.1-SNAPSHOT</version>
<executions>
<execution>
<phase>package</phase>
<goals>
<goal>springFactoriesMerge</goal>
</goals>
</execution>
</executions>
<configuration>
<factoriesBaseClassPathDir>${basedir}</factoriesBaseClassPathDir>
<finalJarName>${project.artifactId}-${project.version}</finalJarName>
</configuration>
</plugin>
</plugins>
</build>這邊有個小細節(jié)是當maven-shade-plugin和spring-factories-merge-plugin的執(zhí)行生命周期都是相同階段,比如都是在package時,則maven-shade-plugin放置順序得在spring-factories-merge-plugin之前,因為spring-factories-merge-plugin是對maven-shade-plugin打包后的結(jié)果進行二次加工。如果maven-shade-plugin不放置順序得在spring-factories-merge-plugin之前,則spring-factories-merge-plugin的執(zhí)行階段就要比maven-shade-plugin靠后,比如maven-shade-plugin在package階段執(zhí)行,則spring-factories-merge-plugin就得在install或者deploy階段執(zhí)行
打包后的效果圖如下

總結(jié)
之前在看開源框架的時候,很經(jīng)常都是聚焦在源碼上,而不會去注意一些maven插件,這次因為有這打jar的需求。我發(fā)現(xiàn)不管是springboot還是dubbo本身就集成一些寶藏插件,比如這個maven-shade-plugin插件,我就是dubbo那邊找到的,地址在
https://github.com/apache/dubbo/blob/master/dubbo-all/pom.xml。比如版本占位符插件flatten-maven-plugin在dubbo和springboot都有看到使用。如果后面有對maven插件由需求,推薦可以從springboot或者dubbo那邊去搜,估計會有意想不到的收獲
demo鏈接 https://github.com/lyb-geek/springboot-learning/tree/master/springboot-jar-merge
以上就是避免多個jar通過maven打包導(dǎo)致同名配置文件覆蓋沖突問題的詳細內(nèi)容,更多關(guān)于maven打包jar同名沖突的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
SpringSecurity OAuth2單點登錄和登出的實現(xiàn)
本文主要介紹了SpringSecurity OAuth2單點登錄和登出的實現(xiàn),文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-02-02
Java動態(tài)規(guī)劃之丑數(shù)問題實例講解
這篇文章主要介紹了Java動態(tài)規(guī)劃之丑數(shù)問題實例,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)吧2022-09-09
使用EasyPOI實現(xiàn)多sheet動態(tài)列導(dǎo)出
這篇文章主要為大家詳細介紹了如何使用EasyPOI根據(jù)指定時間范圍創(chuàng)建動態(tài)列,以及如何將數(shù)據(jù)組織成符合要求的格式并導(dǎo)出,感興趣的可以了解下2025-03-03
SpringBoot獲取ApplicationContext的3種方式
這篇文章主要為大家詳細介紹了SpringBoot獲取ApplicationContext的3種方式,具有一定的參考價值,感興趣的小伙伴們可以參考一下2019-09-09
BeanUtils.copyProperties()屬性名相同但是類型不同問題
這篇文章主要介紹了BeanUtils.copyProperties()屬性名相同但是類型不同問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-09-09
Springcloud-nacos實現(xiàn)配置和注冊中心的方法
這篇文章主要介紹了Springcloud-nacos實現(xiàn)配置和注冊中心的方法,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-07-07
Java 多線程Synchronized和Lock的區(qū)別
這篇文章主要介紹了Java 多線程Synchronized和Lock的區(qū)別,幫助大家更好的理解和使用Java,感興趣的朋友可以了解下2021-01-01
Java接口的作用_動力節(jié)點Java學(xué)院整理
這篇文章主要介紹了Java接口的作用,涉及到接口的規(guī)范相關(guān)知識,需要的的朋友參考下2017-04-04

