淺談Maven包沖突的原理及解決方法
1.概述
Apache Maven ,是一個軟件(特別是Java軟件)項目管理及自動構(gòu)建工具。在沒有Maven的上古年代,項目中引入jar包需要手動下載一個個的去下載,但是隨著代碼數(shù)量的增加,引入的jar包數(shù)量自然會增加,隨之而來的就是jar包沖突的問題了。
2.產(chǎn)生jar包沖突的原因
眾所周知,一個項目中不能存在兩個全限定類名一致的Class類,并且jar包的本質(zhì)就是打包好的Class類文件,例如: 將 junit-jupiter-api-5.6.2.jar 文件解壓后,

可以得到多個Class文件,所以項目中同樣不能存在兩個名稱項目的jar包。
與此同時,jar包之間也會存在相互依賴,就拿這個 junit-jupiter-api-5.6.2.jar 舉例: pom坐標為:
<dependency> <groupId>org.junit.jupiter</groupId> <artifactId>junit-jupiter-api</artifactId> <version>5.6.2</version> <scope>test</scope> </dependency>
那么可以在maven中央倉庫找到該jar包的詳細信息,地址: https://repo1.maven.org/maven2/org/junit/jupiter/junit-jupiter-api/5.6.2/

打開.pom文件可以看到該jar包還引用了哪些其他jar包,如下圖所示:

當然我們可以通過IDEAL及maven工具查看完整的依賴樹:

或者將依賴樹信息導出到本地:
mvn dependency:tree > a.txt
如下圖所示:

所以說,隨著我們項目的逐漸龐大,所引入的jar包文件逐漸增多,產(chǎn)生包沖突的可能性也會越來越大,我們不可能都用肉眼去查找項目中的包沖突問題。當然我們可以通過一些工具插件幫助我們查找項目中的jar包沖突,比如說 Maven Helper。
3.引入插件解決沖突
以這個項目為例:

依賴關系圖如下:

我們知道,當兩個jar包產(chǎn)生沖突時,取舍原則是誰離的項目近選擇誰,所以spring-web最終會選擇5.1.8版本的。從最終的啟動命令中也可以看出:
"D:\software\IDEA IU\IntelliJ IDEA 2019.3\jbr\bin\java.exe" "-javaagent:D:\software\IDEA IU\IntelliJ IDEA 2019.3\lib\idea_rt.jar=50098:D:\software\IDEA IU\IntelliJ IDEA 2019.3\bin" -Dfile.encoding=UTF-8 -classpath D:\GitHub_Item\resolve-package-conflict\target\classes;
C:\Users\DELL\.m2\repository\org\springframework\spring-web\5.1.8.RELEASE\spring-web-5.1.8.RELEASE.jar;
C:\Users\DELL\.m2\repository\org\springframework\spring-beans\5.1.8.RELEASE\spring-beans-5.1.8.RELEASE.jar;
C:\Users\DELL\.m2\repository\org\springframework\spring-core\5.1.8.RELEASE\spring-core-5.1.8.RELEASE.jar;
C:\Users\DELL\.m2\repository\org\springframework\spring-jcl\5.1.8.RELEASE\spring-jcl-5.1.8.RELEASE.jar;
C:\Users\DELL\.m2\repository\com\github\hcsp\test-library-a\0.4\test-library-a-0.4.jar Main
使用Maven Helper插件分析:

根據(jù)自己的需要,選擇要排除的版本:

插件就會幫我們在pom.xml文件中,排除掉產(chǎn)生沖突的引入:

點擊 Reimport
按鈕,就可以看到?jīng)_突已經(jīng)解決了。
4.引深學習
4.1 <scope>test</scope> 與 <scope>compile</scope> 的區(qū)別?
對于test而言,表示該依賴只作用于測試類中,也就是src/main/test路徑下,在其他路徑中,編譯器是不會引入該依賴的。對于compile則沒有限制,在src/main/java于src/mian/test中均可用。
4.2 <scope>provided</scope> 表示什么?
舉例說明:
public static void main(String[] args) throws IOException {
Workbook workbook = new HSSFWorkbook(new FileInputStream("C:\\Users\\DELL\\Desktop\\new.xlsx"));
}
pom.xml中引入必要的依賴:
<dependency> <groupId>org.apache.poi</groupId> <artifactId>poi</artifactId> <version>4.1.2</version> <scope>provided</scope> </dependency>
但是在帶點擊允許的時候,就是會報出:
Exception in thread "main" java.lang.NoClassDefFoundError: org/apache/poi/hssf/usermodel/HSSFWorkbook
at Main.main(Main.java:10)
Caused by: java.lang.ClassNotFoundException: org.apache.poi.hssf.usermodel.HSSFWorkbook
at java.base/jdk.internal.loader.BuiltinClassLoader.loadClass(BuiltinClassLoader.java:581)
at java.base/jdk.internal.loader.ClassLoaders$AppClassLoader.loadClass(ClassLoaders.java:178)
at java.base/java.lang.ClassLoader.loadClass(ClassLoader.java:521)
... 1 more
類沒有找到的問題,原因在于設置 <scope>provided</scope> 則表示該依賴只在編譯時的CLASSPATH中,在運行時則不將該依賴加入CLASSPATH中。一半用于運行環(huán)境已經(jīng)將CLASSPATH設置好,不需要額外添加的情況,比如Tomcat。
5.資源共享
1.《Maven實戰(zhàn)》---下載地址:https://github.com/lxw420302/books/blob/master/java/Maven%E5%AE%9E%E6%88%98.pdf
到此這篇關于淺談Maven包沖突的原理及解決方法的文章就介紹到這了,更多相關Maven包沖突內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
Springboot 整合RabbitMq(用心看完這一篇就夠了)
這篇文章主要介紹了Springboot 整合RabbitMq(用心看完這一篇就夠了),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-12-12
SpringMVC空指針異常NullPointerException解決及原理解析
這篇文章主要介紹了SpringMVC空指針異常NullPointerException解決及原理解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-08-08
springboot基于IDEA環(huán)境熱加載與熱部署教程
這篇文章主要為大家介紹了springboot在IDEA環(huán)境下的熱加載與熱部署教程,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步2022-03-03
SpringBoot實現(xiàn)本地上傳文件到resources目錄
Java后端項目上傳文件是一個很常見的需求,這篇文章主要為大家介紹了SpringBoot如何實現(xiàn)本地上傳文件到resources目錄永久保存下載,需要的可以參考一下2023-07-07

