聊聊Maven的依賴傳遞、依賴管理、依賴作用域

来源:https://www.cnblogs.com/zwwhnly/archive/2023/10/12/17758894.html
-Advertisement-
Play Games

1. 依賴傳遞 在Maven中,依賴是會傳遞的,假如在業務項目中引入了spring-boot-starter-web依賴: <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter ...


1. 依賴傳遞

在Maven中,依賴是會傳遞的,假如在業務項目中引入了spring-boot-starter-web依賴:

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-web</artifactId>
  	<version>2.7.4</version>
</dependency>

那麼業務項目不僅直接引入了spring-boot-starter-web依賴,還間接引入了spring-boot-starter-web的依賴項:

spring-boot-starterspring-boot-starter-jsonspring-boot-starter-tomcatspring-webspring-webmvc

Maven依賴關係如下圖所示:

外部庫如下圖所示:

其中,業務項目對spring-boot-starter-web的依賴稱為直接依賴,對spring-boot-starter-web的依賴項:

spring-boot-starterspring-boot-starter-jsonspring-boot-starter-tomcatspring-webspring-webmvc

的依賴稱為間接依賴。

2. 依賴管理

dependencyManagement元素主要用來統一管理依賴項的版本號。

假如父項目的pom文件中聲明瞭如下依賴:

<dependencyManagement>
    <dependencies>
        <dependency>
            <groupId>org.apache.commons</groupId>
            <artifactId>commons-collections4</artifactId>
            <version>4.4</version>
        </dependency>
    </dependencies>
</dependencyManagement>    

那麼子項目在添加該依賴時,可以不指定版本號:

<dependencies>
		<dependency>
        <groupId>org.apache.commons</groupId>
        <artifactId>commons-collections4</artifactId>
    </dependency>
</dependencies>

Maven會自動找到父項目中聲明的該依賴項的版本號,如下圖所示:

這樣的優點是可以統一在父項目中管理依賴項的版本號,如果需要升級版本,只需改動父項目一個地方即可,子項目不用改動。

說明:

1)dependencyManagement只是聲明依賴項,並沒引入依賴項,子項目仍需顯式引入,不過可以不指定版本號

2)如果子項目不想使用繼承的父項目中的版本號,在子項目中指定版本號即可

3. 依賴作用域

在Maven中,可以使用scope來指定當前依賴項的作用域,常見的值有:compile、provided、runtime、test、import等,如下所示:

<dependency>
    <groupId>junit</groupId>
    <artifactId>junit</artifactId>
    <scope>test</scope>
</dependency>

3.1 compile

compile是預設的作用域,如果引入依賴時,沒有明確指定作用域,則依賴作用域為compile。

作用域為compile的依賴,在編譯、測試和運行時都是可用的,並且會參與項目的打包過程,該依賴會傳遞給依賴該模塊的其他模塊。

3.2 provided

作用域為provided的依賴,在編譯和測試時是可用的,在運行時是不可用的,不會參與項目的打包過程,也不會傳遞給其他模塊。

比如lombok依賴會在編譯時生成相應的get、set等方法,在運行時就不需要這個依賴了,因此常常被指定為provided:

<dependency>
    <groupId>org.projectlombok</groupId>
    <artifactId>lombok</artifactId>
    <version>1.18.16</version>
    <scope>provided</scope>
</dependency>

因為被指定為provided,項目打包時是不包含lombok依賴項的,如下圖所示:

如果將上面的代碼<scope>provided</scope>刪除的話,運行時是下圖這樣的:

以上驗證需將項目打包方式改為war,打包後查看WEB-INF/lib目錄

3.3 runtime

作用域為runtime的依賴,在測試和運行時是可用的,在編譯時是不可用的,會參與項目的打包過程,也會傳遞給依賴該模塊的

其他模塊。

說明:

作用域為runtime的依賴中的類,在項目代碼里不能直接用,用了無法通過編譯(這裡指的是在src/main/java下使用)。

以mysql-connector-java為例,假如引入依賴時是下麵這樣的:

<dependency>
    <groupId>mysql</groupId>
    <artifactId>mysql-connector-java</artifactId>
    <version>8.0.30</version>
</dependency>

下麵的示例代碼是可以編譯通過的:

如果將作用域修改為runtime,上面的示例代碼無法通過編譯:

<dependency>
    <groupId>mysql</groupId>
    <artifactId>mysql-connector-java</artifactId>
    <version>8.0.30</version>
    <scope>runtime</scope>
</dependency>

3.4 test

作用域為test的依賴,只在測試時可用(包括測試代碼的編譯、執行),不會參與項目的打包過程,也不會傳遞給其他模塊。

常見的有junit、mockito等:

<dependency>
    <groupId>junit</groupId>
    <artifactId>junit</artifactId>
    <version>4.12</version>
    <scope>test</scope>
</dependency>

因為被指定為test,項目打包時是不包含junit依賴項的,如下圖所示:

如果將上面的代碼<scope>test</scope>刪除的話,運行時是下圖這樣的:

以上驗證需將項目打包方式改為war,打包後查看WEB-INF/lib目錄

說明:

作用域為test的依賴中的類或者註解只能在src/test/java下才可以使用,在src/main/java下無法使用,如junit包下的@Test註解和org.junit.Assert斷言類。

3.5 import

每個項目,一般都會繼承自一個父項目,在實際的工作中,這個父項目一般都是公司架構組提供的帶有公司特色的一個基礎項目,

當然也可以是spring boot官方的項目。

以spring boot官方的項目為例:

<parent>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-parent</artifactId>
    <version>2.7.4</version>
</parent>

這個父項目中,會使用dependencyManagement標簽對依賴項的版本統一管理,子項目中,可以按需引入父項目

dependencyManagement中定義的依賴,但可以不指定版本號(版本號會自動繼承父項目中定義的版本號)。

但是存在以下2個問題:

  1. Maven是單繼承的,一個項目只能有一個parent項目
  2. parent項目dependencyManagement中的依賴項會越來越多,不好管理

依賴作用域import的出現就是為瞭解決以上問題,它可以通過非繼承的方式批量引入另一個依賴項中

dependencyManagement元素中定義的依賴項,如下所示:

<dependencyManagement>
    <dependency>
      <groupId>org.springframework.session</groupId>
      <artifactId>spring-session-bom</artifactId>
      <version>${spring-session-bom.version}</version>
      <type>pom</type>
      <scope>import</scope>
    </dependency>
  </dependencies>
</dependencyManagement>

說明:<scope>import</scope>只能用在dependencyManagement下type為pom的dependency中。

以上代碼等價於添加了以下6個依賴項:

可以看出,使用<scope>import</scope>可以模塊化的管理依賴項,提高復用性,pom文件也更加簡潔。

3.6 區別

綜上所述,各個依賴作用域的區別如下表所示:

scope取值 編譯時可用 測試時可用 運行時可用 是否參與打包 依賴傳遞
compile
provided × × ×
runtime ×
test × × × ×

4. 影響依賴傳遞的因素

4.1 依賴作用域(scope)

依賴作用域會影響依賴傳遞,從上表可以看出,如果scope為provided或者test,該依賴不會傳遞,只有scope為compile或者runtime,

該依賴才會傳遞。

4.2 可選依賴(optional)

通過dependency標簽引入依賴時,可以通過<optional>指定該依賴是不是可選的,預設值為false:

<dependency>
    <groupId>org.ehcache</groupId>
    <artifactId>ehcache</artifactId>
	  <version>3.2.3</version>
    <optional>true</optional>
</dependency>

如果<optional>值為true,那麼這個依賴不會傳遞。


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

-Advertisement-
Play Games
更多相關文章
  • 除了Vue 3這個庫,還需Vue 3 最新全家桶。 1 環境準備 之前語法演示直接使用script引入Vue 3,從而在瀏覽器里實現所有調試功能。但實際項目中,使用專門調試工具。在項目上線之前,代碼也需打包壓縮,並考慮到研發效率和代碼可維護性,所以在下麵,需建立一個工程化項目實現這些功能。 工具 V ...
  • 出於安全的考慮,瀏覽器會禁止 Ajax 訪問不同域的地址,在現如今微服務橫行的年代,跨域訪問是非常常見的。W3C 的 CORS (Cross-origin-resource-sharing) 規範中也已經允許跨域訪問, 並被主流瀏覽器所支持,它們包括: Chrome 3+; Firefox 3.5+ ...
  • 分散式架構的演進 在軟體行業,一個應用服務隨著功能越來越複雜,用戶量越來越大,尤其是互聯網行業流量爆髮式的增長,導致我們需要不斷的重構應用的結構來支撐龐大的用戶量,最終從一個簡單的系統主鍵演變成了一個非常複雜的可以支撐高併發的高可用的分散式架構,但是一個系統再複雜也是不斷演變來的,所以從另一方面來說 ...
  • 前置條件: 適用於常規請求都沒問題,但是執行某些php腳本需要超過一分鐘的情況下的502/504,並不是任何請求都502/504的情況(這說明php-fpm或者nginx配置錯誤)。 出現502/504的原因 502 執行腳本時間太長,期間php沒有返回任何的數據。php-fpm超時,nginx沒超 ...
  • 目錄定義使用條件使用方式匿名類lambda 表達式::引用已有方法總結 定義 將方法作為參數傳遞即為方法引用。 以@FunctionalInterface 修飾的介面 其中只能有一個抽象方法。 這個抽象方法的形參和返回值便是對所引用方法的約束。 使用條件 引用方法的返回值類型和形參需要與函數式介面的 ...
  • 本文深入探討了Go語言中介面的概念和實際應用場景。從基礎知識如介面的定義和實現,到更複雜的實戰應用如解耦與抽象、多態、錯誤處理、插件架構以及資源管理,文章通過豐富的代碼示例和詳細的解釋,展示了Go介面在軟體開發中的強大功能和靈活性。 關註【TechLeadCloud】,分享互聯網架構、雲服務技術的全 ...
  • BeanPostProcessor後置處理器是Spring提供的一個擴展點,可以在Bean初始化前後做一些事情,註意這裡是bean的初始化,不是實例化,BeanPostProcessor是一個介面,裡面提供了兩個方法,分別為postProcessBeforeInitialization(初始化之前) ...
  • 之前,我們通過一系列文章,介紹瞭如何在Spring Boot中發送郵件: 發送郵件 添加附件 引用靜態資源 郵件模版 已經包含了大部分的應用場景。但最近DD在做YouTube中文配音的時候,碰到一個問題: 如上圖所示,收件人在客戶端收到的時候,顯示的名稱是郵箱的首碼,而不是我們的產品名稱,也就是郵箱 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...