Maven 一個(gè)核心的特性就是依賴管理。當(dāng)我們處理多模塊的項(xiàng)目(包含成百上千個(gè)模塊或者子項(xiàng)目),模塊間的依賴關(guān)系就變得非常復(fù)雜,管理也變得很困難。針對(duì)此種情形,Maven 提供了一種高度控制的方法。
一種相當(dāng)常見的情況,比如說 A 依賴于其他庫 B。如果,另外一個(gè)項(xiàng)目 C 想要使用 A ,那么 C 項(xiàng)目也需要使用庫 B。
Maven 可以避免去搜索所有所需庫的需求。Maven 通過讀取項(xiàng)目文件(pom.xml),找出它們項(xiàng)目之間的依賴關(guān)系。
我們需要做的只是在每個(gè)項(xiàng)目的 pom 中定義好直接的依賴關(guān)系。其他的事情 Maven 會(huì)幫我們搞定。
通過可傳遞性的依賴,所有被包含的庫的圖形會(huì)快速的增長。當(dāng)有重復(fù)庫時(shí),可能出現(xiàn)的情形將會(huì)持續(xù)上升。Maven 提供一些功能來控制可傳遞的依賴的程度。
功能 | 功能描述 |
---|---|
依賴調(diào)節(jié) | 決定當(dāng)多個(gè)手動(dòng)創(chuàng)建的版本同時(shí)出現(xiàn)時(shí),哪個(gè)依賴版本將會(huì)被使用。 如果兩個(gè)依賴版本在依賴樹里的深度是一樣的時(shí)候,第一個(gè)被聲明的依賴將會(huì)被使用。 |
依賴管理 | 直接的指定手動(dòng)創(chuàng)建的某個(gè)版本被使用。例如當(dāng)一個(gè)工程 C 在自己的依賴管理模塊包含工程 B,即 B 依賴于 A, 那么 A 即可指定在 B 被引用時(shí)所使用的版本。 |
依賴范圍 | 包含在構(gòu)建過程每個(gè)階段的依賴。 |
依賴排除 | 任何可傳遞的依賴都可以通過 "exclusion" 元素被排除在外。舉例說明,A 依賴 B, B 依賴 C,因此 A 可以標(biāo)記 C 為 "被排除的"。 |
依賴可選 | 任何可傳遞的依賴可以被標(biāo)記為可選的,通過使用 "optional" 元素。例如:A 依賴 B, B 依賴 C。因此,B 可以標(biāo)記 C 為可選的, 這樣 A 就可以不再使用 C。 |
傳遞依賴發(fā)現(xiàn)可以通過使用如下的依賴范圍來得到限制:
范圍 | 描述 |
---|---|
編譯階段 | 該范圍表明相關(guān)依賴是只在項(xiàng)目的類路徑下有效。默認(rèn)取值。 |
供應(yīng)階段 | 該范圍表明相關(guān)依賴是由運(yùn)行時(shí)的 JDK 或者 網(wǎng)絡(luò)服務(wù)器提供的。 |
運(yùn)行階段 | 該范圍表明相關(guān)依賴在編譯階段不是必須的,但是在執(zhí)行階段是必須的。 |
測(cè)試階段 | 該范圍表明相關(guān)依賴只在測(cè)試編譯階段和執(zhí)行階段。 |
系統(tǒng)階段 | 該范圍表明你需要提供一個(gè)系統(tǒng)路徑。 |
導(dǎo)入階段 | 該范圍只在依賴是一個(gè) pom 里定義的依賴時(shí)使用。同時(shí),當(dāng)前項(xiàng)目的POM 文件的 部分定義的依賴關(guān)系可以取代某特定的 POM。 |
通常情況下,在一個(gè)共通的項(xiàng)目下,有一系列的項(xiàng)目。在這種情況下,我們可以創(chuàng)建一個(gè)公共依賴的 pom 文件,該 pom 包含所有的公共的依賴關(guān)系,我們稱其為其他子項(xiàng)目 pom 的 pom 父。 接下來的一個(gè)實(shí)例可以幫助你更好的理解這個(gè)概念。
接下來是上面依賴圖的詳情說明:
App-UI-WAR 依賴于 App-Core-lib 和 App-Data-lib。
Root 是 App-Core-lib 和 App-Data-lib 的父項(xiàng)目。
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>
現(xiàn)在當(dāng)我們構(gòu)建 App-UI-WAR 項(xiàng)目時(shí), Maven 將通過遍歷依賴關(guān)系圖找到所有的依賴關(guān)系,并且構(gòu)建該應(yīng)用程序。
通過上面的實(shí)例,我們可以學(xué)習(xí)到以下關(guān)鍵概念:
公共的依賴可以使用 pom 父的概念被統(tǒng)一放在一起。App-Data-lib 和 App-Core-lib 項(xiàng)目的依賴在 Root 項(xiàng)目里列舉了出來(參考 Root 的包類型,它是一個(gè) POM).
沒有必要在 App-UI-W 里聲明 Lib1, lib2, Lib3 是它的依賴。 Maven 通過使用可傳遞的依賴機(jī)制來實(shí)現(xiàn)該細(xì)節(jié)。