maven依赖版本没有按照最短路径原则生效的解决方案
作者:干货满满张哈希 时间:2023-04-04 20:49:09
女朋友他们项目用了 spring-boot
,以 spring-boot-parent
作为 parent:
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>2.0.9</version>
</parent>
女朋友最近想用 elasticsearch
作为搜索引擎,在项目中添加了依赖
<dependency>
<groupId>org.elasticsearch</groupId>
<artifactId>elasticsearch</artifactId>
<version>7.10.2</version>
</dependency>
写好代码,一跑,报类不存在异常:
java.lang.NoClassDefFoundError: org/elasticsearch/common/xcontent/DeprecationHandler
at com.lv.springboot.datasource.ClientUTis.main(ClientUTis.java:13)
Caused by: java.lang.ClassNotFoundException: org.elasticsearch.common.xcontent.DeprecationHandler
at java.net.URLClassLoader.findClass(URLClassLoader.java:381)
at java.lang.ClassLoader.loadClass(ClassLoader.java:424)
at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:331)
at java.lang.ClassLoader.loadClass(ClassLoader.java:357)
... 1 more
女朋友看了依赖mvn dependency:tree
,发现依赖的elasticsearch
版本是:
org.elasticsearch.client:elasticsearch-rest-high-level-client:7.0.1
|--org.elasticsearch:elasticsearch:5.6.16
|--org.elasticsearch.client:elasticsearch-rest-client:7.0.1
|--org.elasticsearch.plugin:parent-join-client:7.0.1
|--org.elasticsearch.plugin:aggs-matrix-stats-client:7.0.1
|--org.elasticsearch.plugin:rank-eval-client:7.0.1
|--org.elasticsearch.plugin:lang-mustache-client:7.0.1
女朋友很着急,明明指定了elasticsearch
的依赖了啊,而且是项目的根 pom,依赖不是最短路径原则么?不应该以这个依赖为准么?
女朋友于是找我求助,本着面向“对象”,我立马放下手头工作帮忙查看。仔细一看,原来SpringBoot的DependencyManagement中,org.elasticsearch:elasticsearch
已经被包含了(以下为节选):
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-dependencies</artifactId>
<version>2.0.9.RELEASE</version>
<properties>
<elasticsearch.version>5.6.16</elasticsearch.version>
</properties>
<dependencyManagement>
<dependencies>
<dependency>
<groupId>org.elasticsearch</groupId>
<artifactId>elasticsearch</artifactId>
<version>${elasticsearch.version}</version>
</dependency>
</dependencies>
</dependencyManagement>
spring-boot 其实已经考虑到用户可能要换版本了,所以将版本放入了 <properties/>
,properties 也具有最短路径原则,所以可以通过在你的项目根 pom 中的 properties 增加相同 key 修改版本:
<properties>
<elasticsearch.version>7.10.2</elasticsearch.version>
</properties>
所有可以这么替换的属性, spring-boot 官方文档已经列出了,参考官方文档附录:Version Properties
也可以通过 dependencyManagement 的最短路径原则,通过在你的项目根 pom 中的增加想修改依赖的 dependencyManagement 即可:
<dependencyManagement>
<dependencies>
<dependency>
<groupId>org.elasticsearch</groupId>
<artifactId>elasticsearch</artifactId>
<version>7.10.2</version>
</dependency>
</dependencies>
</dependencyManagement>
最后,可以记住下面的原则,就知道项目的依赖到底是哪个版本啦:
Maven依赖可以分为如下几部分:
直接依赖,就是本项目 dependencies 部分的依赖
间接依赖,就是本项目 dependencies 部分的依赖所包含的依赖
依赖管理,就是本项目 dependency management 里面的依赖
parent 的直接依赖
parent 的间接依赖
parent 的依赖管理
bom 的直接依赖(一般没有)
bom 的间接依赖(一般没有)
bom 的依赖管理
可以这么理解依赖:
1.首先,将 parent 的直接依赖,间接依赖,还有依赖管理,插入本项目,放入本项目的直接依赖,间接依赖还有依赖管理之前
2.对于直接依赖,如果有 version,那么就依次放入 DependencyMap 中。如果没有 version,则从依赖管理中查出来 version,之后放入 DependencyMap 中。key 为依赖的 groupId + artifactId,value为version,后放入的会把之前放入的相同 key 的 value 替换
3.对于每个依赖,各自按照 1,2 加载自己的 pom 文件,但是如果第一步中的本项目 dependency management 中有依赖的版本,使用本项目 dependency management的依赖版本,生成 TransitiveDependencyMap,这里面就包含了所有的间接依赖。
4.所有间接依赖的 TransitiveDependencyMap, 对于项目的 DependencyMap 里面没有的 key,依次放入项目的 DependencyMap
5.如果 TransitiveDependencyMap 里面还有间接依赖,那么递归执行3, 4。
由于是先放入本项目的 DependencyMap,再去递归 TransitiveDependencyMap,这就解释了 maven 依赖的最短路径原则。
Bom 的效果基本和 Parent 一样,只是一般限制中,Bom 只有 dependencyManagement 没有 dependencies
解决了问题并且给妹子梳理明白之后,妹子答应这个月多给我 100 块零用钱啦,开心~~~~~
来源:https://juejin.cn/post/6923017489240555534
![](/images/zang.png)
![](/images/jiucuo.png)
猜你喜欢
java过滤器中Filter的ChainFilter过滤链
![](https://img.aspxhome.com/file/2023/5/72575_0s.png)
Android使用kotlin实现多行文本上下滚动播放
![](https://img.aspxhome.com/file/2023/6/105986_0s.gif)
深入了解Java接口回调机制
![](https://img.aspxhome.com/file/2023/0/59310_0s.png)
教你C#将CSV转为Excel的实现方法
![](https://img.aspxhome.com/file/2023/8/122628_0s.png)
C#泛型概念的简介与泛型的使用
![](https://img.aspxhome.com/file/2023/2/96432_0s.png)
JFileChooser实现对选定文件夹内图片自动播放和暂停播放实例代码
Java中volatile关键字实现原理
![](https://img.aspxhome.com/file/2023/1/63081_0s.png)
android调用国家气象局天气预报接口json数据格式解释
Java编程实现月食简单代码分享
![](https://img.aspxhome.com/file/2023/3/79943_0s.png)
C++普通函数指针与成员函数指针实例解析
Spring Boot整合JWT的实现步骤
![](https://img.aspxhome.com/file/2023/3/132073_0s.png)
Unity UGUI教程之实现滑页效果
![](https://img.aspxhome.com/file/2023/0/104300_0s.gif)
java 服务器接口快速开发之servlet详细教程
![](https://img.aspxhome.com/file/2023/0/116140_0s.jpg)
探讨Java中函数是值传递还是引用传递问题
![](https://img.aspxhome.com/file/2023/1/85831_0s.png)
Java常用集合与原理解析
![](https://img.aspxhome.com/file/2023/3/71603_0s.jpg)
java 2d画图示例分享(用java画图)
Struts2实现对action请求对象的拦截操作方法
![](https://img.aspxhome.com/file/2023/2/70342_0s.jpg)
Java中的Static class详解及实例代码
android开发教程之获取使用当前api的应用程序名称
![](https://img.aspxhome.com/file/2023/3/139523_0s.jpg)