身邊有幾位剛使用Maven的同學(xué)表示——在一個(gè)叫"pom.xml"的文件里聲明一個(gè)依賴就不用去手動(dòng)添加jar了,感覺(jué)這東西和自己手動(dòng)管理依賴沒(méi)太大區(qū)別。 當(dāng)然,并不是這樣,在此記錄dependency那些事兒。
dependency一個(gè)依賴可以按照maven的坐標(biāo)標(biāo)準(zhǔn)進(jìn)行定義。比如:
<dependency> <groupId>javax.servlet</groupId> <artifactId>javax.servlet-api</artifactId> <version>3.1.0</version> <scope>PRovided</scope></dependency>
上面是最常見(jiàn)的坐標(biāo)屬性,偶爾也會(huì)看到有classifier
。下面簡(jiǎn)單說(shuō)明一下標(biāo)簽:
引入Junit依賴時(shí)通常需要聲明test scope:
<dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.11</version> <scope>test</scope></dependency>
大概能猜出是測(cè)試時(shí)使用該依賴,但不完全正確。Maven編譯代碼時(shí)需要使用classpath,但classpath不止一種,而是:
Maven會(huì)根據(jù)需要使用不同的classpath,而scope可以用來(lái)控制依賴與這三種classpath之間的關(guān)系。
通常會(huì)使用前三種。scope不僅用來(lái)控制依賴與classpath之間的關(guān)系,還會(huì)對(duì)依賴的傳遞性產(chǎn)生影響。傳遞性依賴? 比如A依賴B,B依賴C,則A對(duì)于B是直接依賴,對(duì)于C是傳遞性依賴。A對(duì)B、B對(duì)C的依賴范圍決定了A對(duì)C的依賴范圍。如何決定? 下面給出一個(gè)關(guān)系表,垂直表示第一依賴,水平表示第二依賴,交叉單元格為傳遞性依賴。
compile | test | provided | runtime | |
compile | compile | runtime | ||
test | test | test | ||
provided | provided | provided | provided | |
runtime | runtime | runtime |
考慮一下這樣的依賴關(guān)系,A-> C -> D(1.0)和A-> B -> D(2.0)此時(shí)應(yīng)該如何處理? 引入兩種D依賴是不可能的。Maven有依賴調(diào)節(jié)原則:
對(duì)于上面的例子,A-> C -> D(1.0)和A-> B -> D(2.0)的路徑長(zhǎng)度相同,但前者聲明早于后者,因此加入的傳遞性依賴則是D(1.0)。
另外,還需要考慮這樣一個(gè)場(chǎng)景。A依賴B、B依賴X和Y,X和Y都是可選依賴,即<optional>true</optional>
,且4個(gè)都是compile。此時(shí),X和Y則不會(huì)被傳遞,對(duì)A是不可見(jiàn)的。
但依賴調(diào)節(jié)并不解決所有問(wèn)題,我們還需要exclusions來(lái)進(jìn)行排除依賴。例如這樣一個(gè)情況,工程中引入了A,A依賴B,但是B的版本過(guò)舊。此時(shí)可以使用exclusions排除該傳遞性依賴,并顯示聲明一個(gè)最新版本的B依賴。比如這樣:
<dependency> <groupId>com.lowagie</groupId> <artifactId>itext</artifactId> <version>4.2.1</version> <exclusions> <exclusion> <artifactId>bcmail-jdk14</artifactId> <groupId>bouncycastle</groupId> </exclusion> <exclusion> <artifactId>bcprov-jdk14</artifactId> <groupId>bouncycastle</groupId> </exclusion> <exclusion> <artifactId>bctsp-jdk14</artifactId> <groupId>bouncycastle</groupId> </exclusion> </exclusions></dependency>
properties用于統(tǒng)一管理屬性,比如我們引入很多spring framework:
<dependency> <groupId>org.springframework</groupId> <artifactId>spring-web</artifactId> <version>4.0.3 RELEASE</version></dependency><dependency> <groupId>org.springframework</groupId> <artifactId>spring-webmvc</artifactId> <version>4.0.3 RELEASE</version></dependency><dependency> <groupId>org.springframework</groupId> <artifactId>spring-jms</artifactId> <version>4.0.3 RELEASE</version></dependency>
可見(jiàn)版本都是一樣的,想更改版本時(shí)再一個(gè)一個(gè)修改太麻煩。
properties可以解決這一問(wèn)題:
<properties> <spring.version>4.0.3.RELEASE</spring.version></properties>
引入spring可以改為:
<dependency> <groupId>org.springframework</groupId> <artifactId>spring-web</artifactId> <version>${spring.version}</version></dependency><dependency> <groupId>org.springframework</groupId> <artifactId>spring-webmvc</artifactId> <version>${spring.version}</version></dependency><dependency> <groupId>org.springframework</groupId> <artifactId>spring-jms</artifactId> <version>${spring.version}</version></dependency>
這樣確實(shí)簡(jiǎn)介了不少,但你可能仍然討厭XML。
興許,以后我們不會(huì)再用XML寫構(gòu)建文件。我們可能會(huì)用一些插件(比如Polyglot for Maven)或者其他的什么東西(比如Gradle)。
repository可以把Maven的倉(cāng)庫(kù)分為兩種:
Maven尋找一個(gè)dependency時(shí)會(huì)先從本地倉(cāng)庫(kù)查找,如果找不到則在遠(yuǎn)程倉(cāng)庫(kù)查找,發(fā)現(xiàn)則下載到本地倉(cāng)庫(kù)使用。如果都查找失敗,會(huì)提示build failure。
或者,我們也可以把本地的jar放到本地倉(cāng)庫(kù)中:
mvn install:install-file -Dfile=jar包的路徑 -DgroupId=我的groupId -DartifactId=我的artifactId -Dversion=我的version -Dpackaging=jar
本地倉(cāng)庫(kù)默認(rèn)路徑是用戶目錄下的.m2/repository/
;該路徑可以在settings.xml
中修改,比如:
<localRepository> /usr/local/maven/repository</localRepository>
那么遠(yuǎn)程倉(cāng)庫(kù)又在哪里?打開(kāi)$M2_HOME/lib/maven-model-builder-3.2.1.jar里的org/apache/maven/model/pom-4.0.0.xml
看到遠(yuǎn)程倉(cāng)庫(kù)的設(shè)置如下:
<repositories> <repository> <id>central</id> <name>Central Repository</name> <url>http://repo.maven.apache.org/maven2</url> <layout>default</layout> <snapshots> <enabled>false</enabled> </snapshots> </repository> </repositories>
當(dāng)然,我們也可以配置其他遠(yuǎn)程倉(cāng)庫(kù),比如這樣:
<repositories> <repository> <id>opensesame</id> <name>Alibaba OpenSource Repsoitory</name> <url>http://code.alibabatech.com/mvn/releases/</url> <snapshots> <enabled>false</enabled> </snapshots> </repository></repositories>
配置遠(yuǎn)程倉(cāng)庫(kù)時(shí)也需要注意一些選項(xiàng)
daily
interval : X
:每隔X分鐘間差warn
,另外有fail
和ignore
為什么區(qū)分release和snapshot? 不能只通過(guò)版本號(hào)進(jìn)行區(qū)分嗎?
試想一下這樣的場(chǎng)景,假設(shè)有A和B兩個(gè)模塊,A依賴B,且B尚未開(kāi)發(fā)完成。如何讓B模塊每次更新后讓A的開(kāi)發(fā)人員獲取?每次更新后提示A的開(kāi)發(fā)人員從VCS上pull下來(lái)構(gòu)建?或者不停地?fù)Q版本號(hào)? 確實(shí),就算B有了變化,但version也是依賴的標(biāo)識(shí)之一。
如果依賴是snapshot則能解決這樣的問(wèn)題,snapshot發(fā)布時(shí)會(huì)加上一個(gè)時(shí)間戳,每次構(gòu)建A的時(shí)候會(huì)檢查B是否最新,間差更新策略就是上面的updatePolicy
。另外,也可以執(zhí)行mvn clean install-U
強(qiáng)制更新。
遠(yuǎn)程倉(cāng)庫(kù)不都是想訪問(wèn)就訪問(wèn)的,有些倉(cāng)庫(kù)出于安全考慮,需要提供認(rèn)證信息才可以訪問(wèn)。認(rèn)證必須在settings.xml
中設(shè)置,下面是一個(gè)例子:
<servers> <server> <id>server001</id> <username>my_login</username> <passWord>my_password</password> <privateKey>${user.home}/.ssh/id_dsa</privateKey> <passphrase>some_passphrase</passphrase> <filePermissions>664</filePermissions> <directoryPermissions>775</directoryPermissions> <configuration></configuration> </server></servers>
新聞熱點(diǎn)
疑難解答
圖片精選
網(wǎng)友關(guān)注