Maven依賴管理

来源:https://www.cnblogs.com/leometeor/archive/2023/02/13/17116511.html
-Advertisement-
Play Games

本文主要記錄Maven依賴管理中關於依賴傳遞和依賴範圍的知識 Maven項目示例 創建3個maven項目,分配依賴log4j 1.2.12, 1.2.13, 1.2.14版本。 <!--項目1--> <groupId>com.leo</groupId> <artifactId>project1</a ...


本文主要記錄Maven依賴管理中關於依賴傳遞和依賴範圍的知識

Maven項目示例

創建3個maven項目,分配依賴log4j 1.2.12, 1.2.13, 1.2.14版本。

<!--項目1-->
<groupId>com.leo</groupId>
<artifactId>project1</artifactId>
<version>1.0-SNAPSHOT</version>
<dependencies>
	<dependency>
		<groupId>log4j</groupId>
		<artifactId>log4j</artifactId>
		<version>1.2.12</version>
	</dependency>
</dependencies>

<!--項目2-->
<groupId>com.leo</groupId>
<artifactId>project2</artifactId>
<version>1.0-SNAPSHOT</version>
<dependencies>
	<dependency>
		<groupId>log4j</groupId>
		<artifactId>log4j</artifactId>
		<version>1.2.13</version>
	</dependency>
</dependencies>

<!--項目3-->
<groupId>com.leo</groupId>
<artifactId>project3</artifactId>
<version>1.0-SNAPSHOT</version>
<dependencies>
	<dependency>
		<groupId>log4j</groupId>
		<artifactId>log4j</artifactId>
		<version>1.2.14</version>
	</dependency>
</dependencies>

此時三個項目的依賴關係如圖所示,三個項目分別依賴了不同版本的log4j。

image

依賴傳遞

現在我們構造這樣一種情況,project3依賴log4j和junit,project2依賴log4j和project3,maven配置如下:

<!--項目2-->
<groupId>com.leo</groupId>
<artifactId>project2</artifactId>
<version>1.0-SNAPSHOT</version>
<dependencies>
	<dependency>
		<groupId>log4j</groupId>
		<artifactId>log4j</artifactId>
		<version>1.2.13</version>
	</dependency>
	<dependency>
		<groupId>com.leo</groupId>
		<artifactId>project3</artifactId>
		<version>1.0-SNAPSHOT</version>
	</dependency>
</dependencies>

<!--項目3-->
<groupId>com.leo</groupId>
<artifactId>project3</artifactId>
<version>1.0-SNAPSHOT</version>
<dependencies>
	<dependency>
		<groupId>log4j</groupId>
		<artifactId>log4j</artifactId>
		<version>1.2.14</version>
	</dependency>
	<dependency>
		<groupId>junit</groupId>
		<artifactId>junit</artifactId>
		<version>4.12</version>
	</dependency>
</dependencies>

此時project2和project3的依賴關係如圖:

image

此時project2的依賴關係中出現了project3以及project3所依賴的包。對於project2來說,此時不光可以使用project3,也可以使用project3所以來的junit包。

同時我們註意到,project3依賴了1.2.14版本的log4j,這與project2本身所依賴的1.2.13版本衝突了。

此時Maven會使用如下3個規則來選擇哪個包生效:

  1. 依賴層級淺的包會覆蓋依賴層級深的包。(示例中project2的log4j 1.2.13版本的依賴層級為1,1.2.14的依賴層級為2,因此第一層的版本生效)
  2. 同層依賴中,先聲明的包版本生效。

image

  1. 同pom.xml文件中,後聲明的版本生效。

依賴隱藏

如果不希望別人在依賴我的包時知道我的包依賴了哪些其他的包,那麼可以在引用依賴時將其標註為<optional>true</optional>,這樣別人在使用這個包時,就不會看到這個包依賴的包。

<!--項目2-->
<groupId>com.leo</groupId>
<artifactId>project2</artifactId>
<version>1.0-SNAPSHOT</version>
<dependencies>
	<dependency>
		<groupId>log4j</groupId>
		<artifactId>log4j</artifactId>
		<version>1.2.13</version>
	</dependency>
	<dependency>
		<groupId>com.leo</groupId>
		<artifactId>project3</artifactId>
		<version>1.0-SNAPSHOT</version>
	</dependency>
</dependencies>

<!--項目3-->
<groupId>com.leo</groupId>
<artifactId>project3</artifactId>
<version>1.0-SNAPSHOT</version>
<dependencies>
	<dependency>
		<groupId>log4j</groupId>
		<artifactId>log4j</artifactId>
		<version>1.2.14</version>
	</dependency>
	<dependency>
		<groupId>junit</groupId>
		<artifactId>junit</artifactId>
		<version>4.12</version>
		<optional>true</optional>
	</dependency>
</dependencies>

這樣配置後,依賴關係如下圖所示

image

可以看到在project2中是無法看到project3所依賴的junit包的。當然也就無法通過依賴傳遞的方式使用到junit包。

依賴屏蔽

當我們引用別人的包時,別人的包中某些依賴我們不想引入自己的項目,那我們可以使用如下方式排除間接依賴的包。

<!--項目2-->
<groupId>com.leo</groupId>
<artifactId>project2</artifactId>
<version>1.0-SNAPSHOT</version>
<dependencies>
	<dependency>
		<groupId>log4j</groupId>
		<artifactId>log4j</artifactId>
		<version>1.2.13</version>
	</dependency>
	<dependency>
		<groupId>com.leo</groupId>
		<artifactId>project3</artifactId>
		<version>1.0-SNAPSHOT</version>
		<!--排除不想依賴的包-->
		<exclusions>
			<exclusion>
				<groupId>log4j</groupId>
				<artifactId>log4j</artifactId>
			</exclusion>
		</exclusions>
	</dependency>
</dependencies>

<!--項目3-->
<groupId>com.leo</groupId>
<artifactId>project3</artifactId>
<version>1.0-SNAPSHOT</version>
<dependencies>
	<dependency>
		<groupId>log4j</groupId>
		<artifactId>log4j</artifactId>
		<version>1.2.14</version>
	</dependency>
	<dependency>
		<groupId>junit</groupId>
		<artifactId>junit</artifactId>
		<version>4.12</version>
	</dependency>
</dependencies>

如此配置後的依賴關係如圖:

image

可以看到project2中沒有引入project3所依賴的log4j。

依賴範圍

引入依賴時,可以指定依賴包的生效範圍

scope 主代碼 測試代碼 打包 範例
compile(預設) Y Y Y log4j
test Y junit
provided Y Y servlet-api
runtime Y jdbc
  • provided解釋:例如,我們本地開發時使用了servlet-api 3.0版本進行調測,但當我們要把代碼部署到伺服器的tomcat中時,伺服器的tomcat不支持3.0版本的servlet-api,它自帶了自己的版本。如果我們把3.0版本打包進去,由於3.0版本會先載入,就導致在伺服器的tomcat中無法運行。
  • runtime解釋:jdbc平時我們在使用的時候,載入driver都是通過字元串的形式,在代碼中並沒有真正引用過driver的內容,因此開發時可以不用引入需要依賴的driver,只要打包時把driver打進去就行了。

依賴範圍的傳遞性

compile test provided runtime
compile compile test provided runtime
test
provided
runtime runtime test provided runtime

行表示本項目所依賴的包配置的scope,列表示依賴包的依賴包的scope。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 《Terraform 101 從入門到實踐》這本小冊在南瓜慢說官方網站和GitHub兩個地方同步更新,書中的示例代碼也是放在GitHub上,方便大家參考查看。 簡介 Azure是微軟的公有雲,它提供了一些免費的資源,具體可以查看: https://azure.microsoft.com/en-us/ ...
  • 譯者:kefate 原文:https://github.com/google/guice/wiki/Overview 大家好,我是kefate。今天開始我將會把Google Guice的官方文檔陸續翻譯一遍,水平有限,若有翻譯不妥之處,還望各位不吝指出。OK,話不多說,下麵開始今天的正文~ Guic ...
  • ioc,Inversion of Control(控制反轉),是Spring中的一種設計思想而非技術。 我們可以從4個方面理解ioc: ①誰控制誰? ——Ioc容器控制對象。 ②控制了什麼? ——Ioc容器控制了獲取對象及其外部資源。 ③為什麼是反轉? ——傳統的javaSE技術是手動new對象的, ...
  • 這篇文章主要描述分散式系統中的單體調度機制,它是指一個集群中只有一個節點運行調度進程,並介紹Google Borg的單體調度設計思路。 ...
  • SpringMVC底層機制簡單實現-04 https://github.com/liyuelian/springmvc-demo.git 8.任務7-完成簡單視圖解析 功能說明:通過目標方法返回的 String,轉發或重定向到指定頁面 8.1分析 原生的 SpringMVC 使用視圖解析器來對 Ha ...
  • 一、說明 之前項目中一直使用ConcurrentLinkedQueue做為緩衝隊列(主要是單個項目內,單條改批量的場景,多個項目間使用的是rocketmq),雖然用著方便但是是純記憶體的, 如果項目發生異常崩潰記憶體隊列中的數據就會全部丟失(只能從日誌中恢復)。所以一直想找一個簡單高效支持持久化的嵌入式 ...
  • 書接上文,上回說到如何通過interactcli-rs四步實現一個命令行程式。但是 shell 交互模式在有些場景下用戶體驗並不是很好。比如我們要連接某個服務,比如 mysql 或者 redis 這樣的服務。如果每次交互都需要輸入地址、埠、用戶名等信息,交互起來太麻煩。通常的做法是一次性輸入和連接... ...
  • 教程簡介 Microsoft Azure初學者教程 - 從簡單和簡單的步驟學習Microsoft Azure,從基本到高級概念,包括雲計算概述,Windows Azure,組件,計算模塊,結構控制器,存儲,Blob,隊列,表,CDN,應用程式,安全性,數據中心,方案,管理門戶,創建虛擬網路,部署虛擬 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...