maven 依賴管理
maven 一個核心的特性就是依賴管理。當我們處理多模塊的項目(包含成百上千個模塊或者子項目),模塊間的依賴關系就變得非常復雜,管理也變得很困難。針對此種情形,maven 提供了一種高度控制的方法。
1. 可傳遞性依賴發現
一種相當常見的情況,比如說 a 依賴于其他庫 b。如果,另外一個項目 c 想要使用 a ,那么 c 項目也需要使用庫 b。
maven 可以避免去搜索所有所需庫的需求。maven 通過讀取項目文件(pom.xml),找出它們項目之間的依賴關系。
我們需要做的只是在每個項目的 pom 中定義好直接的依賴關系。其他的事情 maven 會幫我們搞定。
通過可傳遞性的依賴,所有被包含的庫的圖形會快速的增長。當有重復庫時,可能出現的情形將會持續上升。maven 提供一些功能來控制可傳遞的依賴的程度。
功能 | 功能描述 |
---|---|
依賴調節 | 決定當多個手動創建的版本同時出現時,哪個依賴版本將會被使用。 如果兩個依賴版本在依賴樹里的深度是一樣的時候,第一個被聲明的依賴將會被使用。 |
依賴管理 | 直接的指定手動創建的某個版本被使用。例如當一個工程 c 在自己的依賴管理模塊包含工程 b,即 b 依賴于 a, 那么 a 即可指定在 b 被引用時所使用的版本。 |
依賴范圍 | 包含在構建過程每個階段的依賴。 |
依賴排除 | 任何可傳遞的依賴都可以通過 "exclusion" 元素被排除在外。舉例說明,a 依賴 b, b 依賴 c,因此 a 可以標記 c 為 "被排除的"。 |
依賴可選 | 任何可傳遞的依賴可以被標記為可選的,通過使用 "optional" 元素。例如:a 依賴 b, b 依賴 c。因此,b 可以標記 c 為可選的, 這樣 a 就可以不再使用 c。 |
2. 依賴范圍
傳遞依賴發現可以通過使用如下的依賴范圍來得到限制:
范圍 | 描述 |
---|---|
編譯階段 | 該范圍表明相關依賴是只在項目的類路徑下有效。默認取值。 |
供應階段 | 該范圍表明相關依賴是由運行時的 jdk 或者 網絡服務器提供的。 |
運行階段 | 該范圍表明相關依賴在編譯階段不是必須的,但是在執行階段是必須的。 |
測試階段 | 該范圍表明相關依賴只在測試編譯階段和執行階段。 |
系統階段 | 該范圍表明你需要提供一個系統路徑。 |
導入階段 | 該范圍只在依賴是一個 pom 里定義的依賴時使用。同時,當前項目的pom 文件的 部分定義的依賴關系可以取代某特定的 pom。 |
3. 依賴管理
通常情況下,在一個共通的項目下,有一系列的項目。在這種情況下,我們可以創建一個公共依賴的 pom 文件,該 pom 包含所有的公共的依賴關系,我們稱其為其他子項目 pom 的 pom 父。 接下來的一個例子可以幫助你更好的理解這個概念。
接下來是上面依賴圖的詳情說明:
app-ui-war 依賴于 app-core-lib 和 app-data-lib。
root 是 app-core-lib 和 app-data-lib 的父項目。
root 在它的依賴部分定義了 lib1、lib2 和 lib3 作為依賴。
app-ui-war 的 pom.xml 文件代碼如下:
<project xmlns="http://maven.apache.org/pom/4.0.0" xmlns:xsi="http://www.w3.org/2001/xmlschema-instance" xsi:schemalocation="http://maven.apache.org/pom/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> <modelversion>4.0.0</modelversion> <groupid>com.companyname.groupname</groupid> <artifactid>app-ui-war</artifactid> <version>1.0</version> <packaging>war</packaging> <dependencies> <dependency> <groupid>com.companyname.groupname</groupid> <artifactid>app-core-lib</artifactid> <version>1.0</version> </dependency> </dependencies> <dependencies> <dependency> <groupid>com.companyname.groupname</groupid> <artifactid>app-data-lib</artifactid> <version>1.0</version> </dependency> </dependencies> </project>
app-core-lib 的 pom.xml 文件代碼如下:
<project xmlns="http://maven.apache.org/pom/4.0.0" xmlns:xsi="http://www.w3.org/2001/xmlschema-instance" xsi:schemalocation="http://maven.apache.org/pom/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> <parent> <artifactid>root</artifactid> <groupid>com.companyname.groupname</groupid> <version>1.0</version> </parent> <modelversion>4.0.0</modelversion> <groupid>com.companyname.groupname</groupid> <artifactid>app-core-lib</artifactid> <version>1.0</version> <packaging>jar</packaging> </project>
app-data-lib 的 pom.xml 文件代碼如下:
<project xmlns="http://maven.apache.org/pom/4.0.0" xmlns:xsi="http://www.w3.org/2001/xmlschema-instance" xsi:schemalocation="http://maven.apache.org/pom/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> <parent> <artifactid>root</artifactid> <groupid>com.companyname.groupname</groupid> <version>1.0</version> </parent> <modelversion>4.0.0</modelversion> <groupid>com.companyname.groupname</groupid> <artifactid>app-data-lib</artifactid> <version>1.0</version> <packaging>jar</packaging> </project>
root 的 pom.xml 文件代碼如下:
<project xmlns="http://maven.apache.org/pom/4.0.0" xmlns:xsi="http://www.w3.org/2001/xmlschema-instance" xsi:schemalocation="http://maven.apache.org/pom/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> <modelversion>4.0.0</modelversion> <groupid>com.companyname.groupname</groupid> <artifactid>root</artifactid> <version>1.0</version> <packaging>pom</packaging> <dependencies> <dependency> <groupid>com.companyname.groupname1</groupid> <artifactid>lib1</artifactid> <version>1.0</version> </dependency> </dependencies> <dependencies> <dependency> <groupid>com.companyname.groupname2</groupid> <artifactid>lib2</artifactid> <version>2.1</version> </dependency> </dependencies> <dependencies> <dependency> <groupid>com.companyname.groupname3</groupid> <artifactid>lib3</artifactid> <version>1.1</version> </dependency> </dependencies> </project>
現在當我們構建 app-ui-war 項目時, maven 將通過遍歷依賴關系圖找到所有的依賴關系,并且構建該應用程序。
通過上面的例子,我們可以學習到以下關鍵概念:
- 公共的依賴可以使用 pom 父的概念被統一放在一起。app-data-lib 和 app-core-lib 項目的依賴在 root 項目里列舉了出來(參考 root 的包類型,它是一個 pom).
- 沒有必要在 app-ui-w 里聲明 lib1, lib2, lib3 是它的依賴。 maven 通過使用可傳遞的依賴機制來實現該細節。