Maven依赖冲突你会怎么解决

直奔主题,课外知识自行脑补。

1、传递依赖和继承依赖

继承依赖是个啥?就是你定义了一个父模块,然后子模块的pom里定义了parent去引用父模块,然后子模块会继承父模块的声明。

再简单多说一句多模块和继承的在编译时的玩法:

1、有依赖关系的多模块项目,工程结构如下图

1)直接编译父项目。maven会通过Reactor机制制定编译顺序(maven会找到各子module的关系),maven知道父模块下的所有子模块,并会按照依赖关系依次处理执行依次编译所有项目。

2)编译maven子项目时,maven不仅对子项目编译执行,也编译执行了父项目,但不会编译兄弟项目及其兄弟项目的子项目。

2、不具有依赖关系的多模块项目

即兄弟项目间互不关联,都只与父项目存在依赖关系。表现就是对于这类型的项目,我们可以删除父模块的...</modules>部分,而子模块保留parent部分。此时项目不再是一个多模块项目,而是多工程项目的一个集合。

1)此时编译父模块就只会编译父模块自己。

2)编译子模块时,会将子模块和父模块一块编译,但不会编译兄弟模块。

3)若保留父模块的<modules>...</modules>部分而删除子模块的parent部分,那么项目不再是父子项目,只是一个多模块项目,每个子项目独享自己的资源,不能共享。

跑题了,接着讲:

一、继承依赖,

只需记住这句:子模块会依赖父模块的声明,若子模块单独声明,就用自己的。

二、传递依赖

假如A依赖B,B依赖C,那么A也依赖C。(也有例外啊,如果B依赖C时指定了optional为true,标记为可选依赖,则A不会依赖C)。

传递性依赖有以下几个规则:

1)         最短路径原则:如果A对于依赖路径中有两个相同的jar包,那么选择路径短的那个包,路径最近者优先,上述会选X(2.0)。

2)         第一声明优先原则:如果A对于依赖路径中有两个相同的jar包,路径长度也相同,那么依赖写在前面的优先。例如:A->B->F(1.0),A->C->F(2.0),会选F(1.0)。

3)         可选依赖不会被传递,如A->B,B->C,B->D,A对B直接依赖,B对C和D是可选依赖,那么在A中不会引入C和D。可选依赖通过optional元素配置,true表示可选。如果要在A项目中使用C或者D则需要显式地声明C或者D依赖。

讲到这里说说今天要说的重点,也是写这篇文章的缘由,Maven dependencyManagement中的依赖版本会覆盖传递依赖版本。

现在有如下工程:maven_dep_demo 是父module,有dao,service,web三个子module,子module的依赖关系dao->service->web.


回答完下面几道题,maven的一些基础依赖,你基本就理解的差不多了,一定要自己下去多试,答题时间开始:

第一题:

1)dao引用了spring-core的5.1.6.

2)service只是引用的dao

3)web 在dependencyManagement中定义了spring-core 的版本是4.3.23

那大家猜一下,在dao,service 和web中显示的spring的版本号是什么?

答案是:

结论:pom文件中没有指定版本的依赖或是传递的依赖,如果在本module中dependencyManagement中有指定此依赖版本,那就使用本module中dependencyManagement中定义的版本号。

第二题:

1)dao引用了spring-core的5.1.6.如上

2)service只是引用的dao,如上

3)web 只是引用了service。

4)父pom中在dependencyManagement中定义了spring-core 的版本是4.3.23.RELEASE


那大家猜一下,在dao,service 和web中显示的spring的版本号是什么? 

结论:如果在dependency中没有指定version,那么就会依次找当前pom中以及父pom中的dependencyManagement是否定义了版本。

第三题:

1)dao引用了spring-core的5.1.6.如上

2)service只是引用的dao,如上

3)web 在dependencyManagement中定义了spring-core 的版本是4.3.23

4)父pom中在dependencyManagement中定义了spring-core 的版本是3.2.9


那大家猜一下,在dao,service 和web中显示的spring的版本号是什么? 

结论:子的永远优先父的。

第四题:

1)我给原来的工程里添加一个module叫common,common继承父pom。common中引用spring3


2)dao引用了spring-core的5.1.6.

3)service只是引用的dao

4)在web里定义dependencyManagement引用common,且web引用service。

那大家猜一下,在dao,service 和web中显示的spring的版本号是什么? 

第五题:

1)我给原来的工程里添加一个module叫common,common继承父pom。common中引用spring3


2)dao引用了spring-core的5.1.6.

3)service只是引用的dao

4)在web引用service。。

5)在父pom里定义dependencyManagement引用common

那大家猜一下,在dao,service 和web中显示的spring的版本号是什么? 

结论:dependencyManagement的定义不具有传递性。

第六题:

1)我给原来的工程里添加一个module叫common,common继承父pom。


common中只是dependencyManagementspring3.2.9

2)dao引用了spring-core的5.1.6.

3)service只是引用的dao

4)在web引用service。。

5)在父pom里空的。

6)web的引用common

那大家猜一下,在dao,service 和web中显示的spring的版本号是什么? 

第七题:

1)我给原来的工程里添加一个module叫common,common继承父pom。


common中dependencyManagement spring3.2.9且依赖。

2)dao引用了spring-core的5.1.6.

3)service只是引用的dao

4)在web引用service。。

5)在父pom里空的。

6)web的引用common

那大家猜一下,在dao,service 和web中显示的spring的版本号是什么? 

dao 和 service 变成了spring5,web中变成了这样


第八题:

1)我给原来的工程里添加一个module叫common,common继承父pom。如上

common中只是dependencyManagement 

2)dao引用了spring-core的5.1.6.

3)service只是引用的dao

4)在web引用service。。

5)在父pom里空的。

6)web的引用common,但是type是pom,scope是import


那大家猜一下,在dao,service 和web中显示的spring的版本号是什么? 

dao 和 service 变成了spring5,web中变成了这样


第九题:

1)我给原来的工程里添加一个module叫common,common继承父pom。如上

common中只是dependencyManagement 

2)dao引用了spring-core的5.1.6.

3)service只是引用的dao

4)在web引用service。。

5)在父pom里dependencyManagement common,但是type是pom,scope是import。

6)web的引用common

那大家猜一下,在dao,service 和web中显示的spring的版本号是什么? 

dao是spring 5, service 和web 变成了spring3,web中变成了这样

结论:import就是把pom的配置导进来。跟你写在里面差不多。

综上所述:

1、pom文件中没有指定版本的依赖或是传递的依赖,如果在本module中dependencyManagement中有指定此依赖版本,那就使用本module中dependencyManagement中定义的版本号。

2、如果在dependency中没有指定version,那么就会依次找当前pom中以及父pom中的dependencyManagement是否定义了版本。

3、子的永远优先父的。

4、dependencyManagement的定义不具有传递性。

5、import就是把pom的配置导进来

6、工具不好用就,就用命令吧maven -X dependency:tree

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。