maven依賴版本沒(méi)有按照最短路徑原則生效的解決方案
女朋友他們項(xiàng)目用了 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 作為搜索引擎,在項(xiàng)目中添加了依賴
<dependency> <groupId>org.elasticsearch</groupId> <artifactId>elasticsearch</artifactId> <version>7.10.2</version> </dependency>
寫好代碼,一跑,報(bào)類不存在異常:
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,發(fā)現(xiàn)依賴的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的依賴了啊,而且是項(xiàng)目的根 pom,依賴不是最短路徑原則么?不應(yīng)該以這個(gè)依賴為準(zhǔn)么?
女朋友于是找我求助,本著面向“對(duì)象”,我立馬放下手頭工作幫忙查看。仔細(xì)一看,原來(lái)SpringBoot的DependencyManagement中,org.elasticsearch:elasticsearch已經(jīng)被包含了(以下為節(jié)選):
<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 其實(shí)已經(jīng)考慮到用戶可能要換版本了,所以將版本放入了 <properties/>,properties 也具有最短路徑原則,所以可以通過(guò)在你的項(xiàng)目根 pom 中的 properties 增加相同 key 修改版本:
<properties> <elasticsearch.version>7.10.2</elasticsearch.version> </properties>
所有可以這么替換的屬性, spring-boot 官方文檔已經(jīng)列出了,參考官方文檔附錄:Version Properties
也可以通過(guò) dependencyManagement 的最短路徑原則,通過(guò)在你的項(xiàng)目根 pom 中的增加想修改依賴的 dependencyManagement 即可:
<dependencyManagement> <dependencies> <dependency> <groupId>org.elasticsearch</groupId> <artifactId>elasticsearch</artifactId> <version>7.10.2</version> </dependency> </dependencies> </dependencyManagement>
最后,可以記住下面的原則,就知道項(xiàng)目的依賴到底是哪個(gè)版本啦:
Maven依賴可以分為如下幾部分:
- 直接依賴,就是本項(xiàng)目 dependencies 部分的依賴
- 間接依賴,就是本項(xiàng)目 dependencies 部分的依賴所包含的依賴
- 依賴管理,就是本項(xiàng)目 dependency management 里面的依賴
- parent 的直接依賴
- parent 的間接依賴
- parent 的依賴管理
- bom 的直接依賴(一般沒(méi)有)
- bom 的間接依賴(一般沒(méi)有)
- bom 的依賴管理
可以這么理解依賴:
1.首先,將 parent 的直接依賴,間接依賴,還有依賴管理,插入本項(xiàng)目,放入本項(xiàng)目的直接依賴,間接依賴還有依賴管理之前
2.對(duì)于直接依賴,如果有 version,那么就依次放入 DependencyMap 中。如果沒(méi)有 version,則從依賴管理中查出來(lái) version,之后放入 DependencyMap 中。key 為依賴的 groupId + artifactId,value為version,后放入的會(huì)把之前放入的相同 key 的 value 替換
3.對(duì)于每個(gè)依賴,各自按照 1,2 加載自己的 pom 文件,但是如果第一步中的本項(xiàng)目 dependency management 中有依賴的版本,使用本項(xiàng)目 dependency management的依賴版本,生成 TransitiveDependencyMap,這里面就包含了所有的間接依賴。
4.所有間接依賴的 TransitiveDependencyMap, 對(duì)于項(xiàng)目的 DependencyMap 里面沒(méi)有的 key,依次放入項(xiàng)目的 DependencyMap
5.如果 TransitiveDependencyMap 里面還有間接依賴,那么遞歸執(zhí)行3, 4。
由于是先放入本項(xiàng)目的 DependencyMap,再去遞歸 TransitiveDependencyMap,這就解釋了 maven 依賴的最短路徑原則。
Bom 的效果基本和 Parent 一樣,只是一般限制中,Bom 只有 dependencyManagement 沒(méi)有 dependencies
解決了問(wèn)題并且給妹子梳理明白之后,妹子答應(yīng)這個(gè)月多給我 100 塊零用錢啦,開(kāi)心~~~~~
以上就是maven依賴版本沒(méi)有生效的解決方案的詳細(xì)內(nèi)容,更多關(guān)于maven依賴版本沒(méi)有生效的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Spring?Boot?接口加解密功能實(shí)現(xiàn)
在我們?nèi)粘5腏ava開(kāi)發(fā)中,免不了和其他系統(tǒng)的業(yè)務(wù)交互,或者微服務(wù)之間的接口調(diào)用;如果我們想保證數(shù)據(jù)傳輸?shù)陌踩瑢?duì)接口出參加密,入?yún)⒔饷?,這篇文章主要介紹了Spring?Boot?接口加解密功能實(shí)現(xiàn),需要的朋友可以參考下2023-04-04
基于Java解決華為機(jī)試實(shí)現(xiàn)整數(shù)與IP地址間的轉(zhuǎn)換?
這篇文章主要介紹了基于Java解決華為機(jī)試實(shí)現(xiàn)整數(shù)與IP地址間的轉(zhuǎn)換,文章舉例說(shuō)明圍繞文章主題展開(kāi)相關(guān)內(nèi)容,具有一定的參考價(jià)值,需要的小伙伴可以參考一下2022-02-02
java如何讀取文件目錄返回樹(shù)形結(jié)構(gòu)
這篇文章主要介紹了java如何讀取文件目錄返回樹(shù)形結(jié)構(gòu)問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-01-01
使用System.exit()來(lái)優(yōu)雅地終止SpringBoot項(xiàng)目的代碼示例
System.exit() 方法是 Java 中用于退出程序的方法,它接受一個(gè)整數(shù)參數(shù),通常被用來(lái)指示程序的退出狀態(tài),本文給大家介紹了如何使用System.exit()來(lái)優(yōu)雅地終止SpringBoot項(xiàng)目,需要的朋友可以參考下2024-08-08
Java 語(yǔ)言中Object 類和System 類詳解
Object 是 Java 類庫(kù)中的一個(gè)特殊類,也是所有類的父類。今天通過(guò)本文給大家介紹java object類的簡(jiǎn)單概念及常用方法,需要的朋友參考下吧2021-07-07
AndroidQ沙盒機(jī)制之分區(qū)存儲(chǔ)適配
這篇文章主要介紹了AndroidQ沙盒機(jī)制之分區(qū)存儲(chǔ)適配,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-06-06

