1. 程式人生 > >maven 倉庫配置 pom中repositories屬性

maven 倉庫配置 pom中repositories屬性

什麼是Maven倉庫
在不用Maven的時候,比如說以前我們用Ant構建專案,在專案目錄下,往往會看到一個名為/lib的子目錄,那裡存放著各類第三方依賴jar檔案,如log4j.jar,junit.jar等等。每建立一個專案,你都需要建立這樣的一個/lib目錄,然後複製一對jar檔案,這是很明顯的重複。重複永遠是噩夢的起點,多個專案不共用相同的jar檔案,不僅會造成磁碟資源的浪費,也使得版本的一致性管理變得困難。此外,如果你使用版本管理工具,如SVN(你沒有使用版本管理工具?馬上試試SVN吧,它能幫你解決很多頭疼的問題),你需要將大量的jar檔案提交到程式碼庫裡,可是版本管理工具在處理二進位制檔案方面並不出色。
Maven倉庫就是放置所有JAR檔案(WAR,ZIP,POM等等)的地方,所有Maven專案可以從同一個Maven倉庫中獲取自己所需要的依賴JAR,這節省了磁碟資源。此外,由於Maven倉庫中所有的JAR都有其自己的座標,該座標告訴Maven它的組ID,構件ID,版本,打包方式等等,因此Maven專案可以方便的進行依賴版本管理。你也不在需要提交JAR檔案到SCM倉庫中,你可以建立一個組織層次的Maven倉庫,供所有成員使用。
簡言之,Maven倉庫能幫助我們管理構件(主要是JAR)。
 
本地倉庫(.m2) vs. 遠端倉庫(聯網)
執行Maven的時候,Maven所需要的任何構件都是直接從本地倉庫獲取的。如果本地倉庫沒有,它會首先嚐試從遠端倉庫下載構件至本地倉庫,然後再使用本地倉庫的構件。
比如說,你的專案配置了junit-3.8的依賴,在你執行mvn test 的時候,Maven需要使用junit-3.8的jar檔案,它首先根據座標查詢本地倉庫,如果找到,就直接使用。如果沒有,Maven會檢查可用的遠端倉庫配置,然後逐個嘗試這些遠端倉庫去下載junit-3.8的jar檔案,如果遠端倉庫存在該檔案,Maven會將其下載到本地倉庫中,繼而使用。如果嘗試過所有遠端倉庫之後,Maven還是沒能夠下載到該檔案,它就會報錯。
Maven預設的本地倉庫地址為${user.home}/.m2/repository 。也就是說,一個使用者會對應的擁有一個本地倉庫。
你也可以自定義本地倉庫的位置,修改${user.home}/.m2/settings.xml :

  <!-- localRepository
   | The path to the local repository maven will use to store artifacts.
   |
   | Default: ${user.home}/.m2/repository-->
  <localRepository>D:/.m2/repository</localRepository>

還可以在執行時指定本地倉庫位置:
mvn clean install -Dmaven.repo.local=/home/juven/myrepo/

還有一點需要理解的是,當我們執行install的時候,Maven實際上是將專案生成的構件安裝到了本地倉庫,也就是說,只有install了之後,其它專案才能使用此專案生成的構件。

接著瞭解一下Maven預設的遠端倉庫,即Maven中央倉庫:
安裝好Maven之後,我們可以建立一個簡單的專案,配置一些簡單的依賴,然後執行mvn clean install,專案就構建好了。我們沒有手工的去下載任何jar檔案,這一切都是因為Maven中央倉庫的存在,當Maven在本地倉庫找不到需要的jar檔案時,它會查詢遠端倉庫,而一個原始的Maven安裝就自帶了一個遠端倉庫——Maven中央倉庫。
這個Maven中央倉庫是在哪裡定義的呢?在我的機器上,我安裝了maven-2.0.10,我可以找到這個檔案:${M2_HOME}/lib/maven-2.0.10-uber.jar ,

但是到了3.xxx版本之後在: maven安裝目錄下的:/lib/maven-model-builder-${version}.jar中

開啟該檔案,能找到超級POM:\org\apache\maven\model\pom-4.0.0.xml ,它是所有Maven POM的父POM,所有Maven專案繼承該配置,你可以在這個POM中發現如下配置:

  <repositories>
    <repository>
      <id>central</id>
      <name>Central Repository</name>
      <url>https://repo.maven.apache.org/maven2</url>
      <layout>default</layout>
      <snapshots>
        <enabled>false</enabled>
      </snapshots>
    </repository>
  </repositories>

關於遠端倉庫的配置,下面的我會詳細解釋,這裡我們只要知道,中央倉庫的id為central,遠端url地址為http://repo.maven.apache.org/maven2,它關閉了snapshot版本構件下載的支援。

POM中配置遠端倉庫
前面我們看到超級POM配置了ID為central的遠端倉庫,我們可以在POM中配置其它的遠端倉庫。這樣做的原因有很多,比如你有一個區域網的遠端倉庫,使用該倉庫能大大提高下載速度,繼而提高構建速度,也有可能你依賴的一個jar在central中找不到,它只存在於某個特定的公共倉庫,這樣你也不得不新增那個遠端倉庫的配置。
這裡我配置一個遠端倉庫指向中央倉庫的中國映象:

<project>
...
  <repositories>
    <repository>
      <id>maven-net-cn</id>
      <name>Maven China Mirror</name>
      <url>http://maven.net.cn/content/groups/public/</url>
      <releases>
        <enabled>true</enabled>
      </releases>
      <snapshots>
        <enabled>false</enabled>
      </snapshots>
    </repository>
  </repositories>
  <pluginRepositories>
    <pluginRepository>
      <id>maven-net-cn</id>
      <name>Maven China Mirror</name>
      <url>http://maven.net.cn/content/groups/public/</url>
      <releases>
        <enabled>true</enabled>
      </releases>
      <snapshots>
        <enabled>false</enabled>
      </snapshots>    
    </pluginRepository>
  </pluginRepositories>
...
</project>

我們先看一下<repositories>的配置,你可以在它下面新增多個<repository> ,每個<repository>都有它唯一的ID,一個描述性的name,以及最重要的,遠端倉庫的url。此外,<releases><enabled>true</enabled></releases>告訴Maven可以從這個倉庫下載releases版本的構件,而<snapshots><enabled>false</enabled></snapshots>告訴Maven不要從這個倉庫下載snapshot版本的構件。禁止從公共倉庫下載snapshot構件是推薦的做法,因為這些構件不穩定,且不受你控制,你應該避免使用。當然,如果你想使用區域網內組織內部的倉庫,你可以啟用snapshot的支援。

關於<repositories>的更詳細的配置及相關解釋,請參考:http://www.sonatype.com/books/maven-book/reference_zh/apas02s08.html。
至於<pluginRepositories>,這是配置Maven從什麼地方下載外掛構件(Maven的所有實際行為都由其外掛完成)。該元素的內部配置和<repository>完全一樣,不再解釋。

settings.xml中配置遠端倉庫
我們知道了如何在POM中配置遠端倉庫,但考慮這樣的情況:在一個公司內部,同時進行這3個專案,而且以後隨著這幾個專案的結束,越來越多的專案會開始;同時,公司內部建立一個Maven倉庫。我們統一為所有這些專案配置該倉庫,於是不得不為每個專案提供同樣的配置。問題出現了,這是重複 !
其實我們可以做到只配置一次,在哪裡配置呢?就是settings.xml。
不過事情沒有那麼簡單,不是簡單的將POM中的<repositories>及<pluginRepositories>元素複製到settings.xml中就可以,setting.xml不直接支援 這兩個元素。但我們還是有一個並不複雜的解決方案,就是利用profile,如下:

<settings>
  ...
  <profiles>
    <profile>
      <id>dev</id>
      <!-- repositories and pluginRepositories here-->
    </profile>
  </profiles>
  <activeProfiles>
    <activeProfile>dev</activeProfile>
  </activeProfiles>
  ...
</settings>

這裡我們定義一個id為dev的profile,將所有repositories以及pluginRepositories元素放到這個profile中,然後,使用<activeProfiles>元素自動啟用該profile。這樣,你就不用再為每個POM重複配置倉庫。
使用profile為settings.xml新增倉庫提供了一種使用者全域性範圍的倉庫配置。

使用映象
如果你的地理位置附近有一個速度更快的central映象,或者你想覆蓋central倉庫配置,或者你想為所有POM使用唯一的一個遠端倉庫(這個遠端倉庫代理的所有必要的其它倉庫),你可以使用settings.xml中的mirror配置。
以下的mirror配置用maven.net.cn覆蓋了Maven自帶的central:

<settings>
...
  <mirrors>
    <mirror>
      <id>maven-net-cn</id>
      <name>Maven China Mirror</name>
      <url>http://maven.net.cn/content/groups/public/</url>
      <mirrorOf>central</mirrorOf>
    </mirror>
  </mirrors>
...
</settings>

這裡唯一需要解釋的是<mirrorOf>,這裡我們配置central的映象,我們也可以配置一個所有倉庫的映象,以保證該映象是Maven唯一使用的倉庫:

  <mirrors>
    <!-- mirror
     | Specifies a repository mirror site to use instead of a given repository. The repository that
     | this mirror serves has an ID that matches the mirrorOf element of this mirror. IDs are used
     | for inheritance and direct lookup purposes, and must be unique across the set of mirrors.
     |
     -->
    <mirror>
      <id>nexus</id>
      <mirrorOf>*</mirrorOf>
      <name>Nexus Mirror</name>
      <url>http://xx.xx/nexus/content/groups/public</url>
    </mirror>
  </mirrors>
<settings>
...
  <mirrors>
    <mirror>
      <id>my-org-repo</id>
      <name>Repository in My Orgnization</name>
      <url>http://192.168.1.100/maven2</url>
      <mirrorOf>*</mirrorOf>
    </mirror>
  </mirrors>
...
</settings>