Maven包衝突的原理及解決方法

1.概述

\quad Apache Maven,是一個軟件(特別是Java軟件)項目管理及自動構建工具。在沒有Maven的上古年代,項目中引入jar包須要手動下載一個個的去下載,可是隨着代碼數量的增長,引入的jar包數量天然會增長,隨之而來的就是jar包衝突的問題了。java

2.產生jar包衝突的緣由

\quad 衆所周知,一個項目中不能存在兩個全限定類名一致的Class類,而且jar包的本質就是打包好的Class類文件,例如: 將junit-jupiter-api-5.6.2.jar文件解壓後,git

能夠獲得多個Class文件,因此項目中一樣不能存在兩個名稱項目的jar包。github

\quad 與此同時,jar包之間也會存在相互依賴,就拿這個junit-jupiter-api-5.6.2.jar舉例: pom座標爲:web

<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/ spring

打開.pom文件能夠看到該jar包還引用了哪些其餘jar包,以下圖所示: apache

\quad 固然咱們能夠經過IDEAL及maven工具查看完整的依賴樹:

或者將依賴樹信息導出到本地:api

mvn dependency:tree > a.txt
複製代碼

以下圖所示:bash

\quad 因此說,隨着咱們項目的逐漸龐大,所引入的jar包文件逐漸增多,產生包衝突的可能性也會愈來愈大,咱們不可能都用肉眼去查找項目中的包衝突問題。固然咱們能夠經過一些工具插件幫助咱們查找項目中的jar包衝突,好比說 Maven Helper

3.引入插件解決衝突

以這個項目爲例:maven

依賴關係圖以下:

咱們知道,當兩個jar包產生衝突時,取捨原則是誰離的項目近選擇誰,因此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插件分析: ide

根據本身的須要,選擇要排除的版本:

插件就會幫咱們在pom.xml文件中,排除掉產生衝突的引入:

點擊 Reimport按鈕,就能夠看到衝突已經解決了。

4.引深學習

4.1 <scope>test</scope><scope>compile</scope>的區別?

\quad 對於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中。通常用於運行環境已經將CLASSPATH設置好,不須要額外添加的狀況,好比Tomcat。

5.資源共享

相關文章
相關標籤/搜索