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.资源共享

相关文章
相关标签/搜索