浅谈Maven包冲突的原理及解决方法

 更新时间:2020年07月20日 15:05:56   作者:我的电脑同志  
这篇文章主要介绍了浅谈Maven包冲突的原理及解决方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

1.概述

Apache Maven ,是一个软件(特别是Java软件)项目管理及自动构建工具。在没有Maven的上古年代,项目中引入jar包需要手动下载一个个的去下载,但是随着代码数量的增加,引入的jar包数量自然会增加,随之而来的就是jar包冲突的问题了。

2.产生jar包冲突的原因

众所周知,一个项目中不能存在两个全限定类名一致的Class类,并且jar包的本质就是打包好的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包文件逐渐增多,产生包冲突的可能性也会越来越大,我们不可能都用肉眼去查找项目中的包冲突问题。当然我们可以通过一些工具插件帮助我们查找项目中的jar包冲突,比如说 Maven Helper

3.引入插件解决冲突

以这个项目为例:

依赖关系图如下:

我们知道,当两个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插件分析:

根据自己的需要,选择要排除的版本:

插件就会帮我们在pom.xml文件中,排除掉产生冲突的引入:

点击 Reimport

按钮,就可以看到冲突已经解决了。

4.引深学习

4.1 <scope>test</scope><scope>compile</scope> 的区别?

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

1.《Maven实战》---下载地址:https://github.com/lxw420302/books/blob/master/java/Maven%E5%AE%9E%E6%88%98.pdf

到此这篇关于浅谈Maven包冲突的原理及解决方法的文章就介绍到这了,更多相关Maven包冲突内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • SpringBoot集成Redis,并自定义对象序列化操作

    SpringBoot集成Redis,并自定义对象序列化操作

    这篇文章主要介绍了SpringBoot集成Redis,并自定义对象序列化操作,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-06-06
  • BCryptPasswordEncoder加密与MD5加密的区别及说明

    BCryptPasswordEncoder加密与MD5加密的区别及说明

    这篇文章主要介绍了BCryptPasswordEncoder加密与MD5加密的区别及说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-08-08
  • 如何根据带账号密码的WSDL地址生成JAVA代码

    如何根据带账号密码的WSDL地址生成JAVA代码

    这篇文章主要介绍了如何根据带账号密码的WSDL地址生成JAVA代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-10-10
  • 解决spring boot hibernate 懒加载的问题

    解决spring boot hibernate 懒加载的问题

    这篇文章主要介绍了解决spring boot hibernate 懒加载的问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-10-10
  • 实现java文章点击量记录实例

    实现java文章点击量记录实例

    这篇文章主要为大家介绍了实现java文章点击量记录实例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-10-10
  • Spring Boot 集成接口管理工具 Knife4j

    Spring Boot 集成接口管理工具 Knife4j

    这篇文章主要介绍了Spring Boot 集成接口管理工具 Knife4j,首先通过创建一个 Spring Boot 项目展开主题,需要的小伙伴可以参考一下
    2022-05-05
  • 关于Sentinel中冷启动限流原理WarmUpController

    关于Sentinel中冷启动限流原理WarmUpController

    这篇文章主要介绍了关于Sentinel中冷启动限流原理WarmUpController,具有很好的参考价值,希望对大家有所帮助。
    2023-04-04
  • Java根据Request获取客户端IP

    Java根据Request获取客户端IP

    这篇文章主要介绍了Java根据Request获取客户端IP的方法,非常不错,具有参考借鉴价值,需要的朋友可以参考下
    2017-05-05
  • java实现MD5加密方法汇总

    java实现MD5加密方法汇总

    本文给大家汇总介绍了2种java实现MD5加密的方法,非常的实用,这里分享给大家,学习下其中的思路,对大家学习java非常有帮助。
    2015-10-10
  • 浅谈JAVA设计模式之享元模式

    浅谈JAVA设计模式之享元模式

    这篇文章主要介绍了JAVA设计模式之享元模式的的相关资料,文中详细的介绍了享元模式的概念以及使用方法,感兴趣的朋友可以了解下
    2020-06-06

最新评论