理解Maven中的SNAPSHOT版本和正式版本

黃博文發表於2016-01-29

Maven中建立的依賴管理方式基本已成為Java語言依賴管理的事實標準,Maven的替代者Gradle也基本沿用了Maven的依賴管理機制。在Maven依賴管理中,唯一標識一個依賴項是由該依賴項的三個屬性構成的,分別是groupId、artifactId以及version。這三個屬性可以唯一確定一個元件(Jar包或者War包)。

其實在Nexus倉庫中,一個倉庫一般分為public(Release)倉和SNAPSHOT倉,前者存放正式版本,後者存放快照版本。如果在專案配置檔案中(無論是build.gradle還是pom.xml)指定的版本號帶有’-SNAPSHOT’字尾,比如版本號為’Junit-4.10-SNAPSHOT’,那麼打出的包就是一個快照版本。

快照版本和正式版本的主要區別在於,本地獲取這些依賴的機制有所不同。假設你依賴一個庫的正式版本,構建的時候構建工具會先在本次倉庫中查詢是否已經有了這個依賴庫,如果沒有的話才會去遠端倉庫中去拉取。所以假設你釋出了Junit-4.10.jar到了遠端倉庫,有一個專案依賴了這個庫,它第一次構建的時候會把該庫從遠端倉庫中下載到本地倉庫快取,以後再次構建都不會去訪問遠端倉庫了。所以如果你修改了程式碼,向遠端倉庫中釋出了新的軟體包,但仍然叫Junit-4.10.jar,那麼依賴這個庫的專案就無法得到最新更新。你只有在重新發布的時候升級版本,比如叫做Junit-4.11.jar,然後通知依賴該庫的專案組也修改依賴版本為Junit-4.11,這樣才能使用到你最新新增的功能。

這種方式在團隊內部開發的時候會變的特別蛋痛。假設有兩個小組負責維護兩個元件,example-service和example-ui,其中example-ui專案依賴於example-service。而這兩個專案每天都會構建多次,如果每次構建你都要升級example-service的版本,那麼你會瘋掉。這個時候SNAPSHOT版本就派上用場了。每天日常構建時你可以構建example-service的快照版本,比如example-service-1.0-SNAPSHOT.jar,而example-ui依賴該快照版本。每次example-ui構建時,會優先去遠端倉庫中檢視是否有最新的example-service-1.0-SNAPSHOT.jar,如果有則下載下來使用。即使本地倉庫中已經有了example-service-1.0-SNAPSHOT.jar,它也會嘗試去遠端倉庫中檢視同名的jar是否是最新的。有的人可能會問,這樣不就不能充分利用本地倉庫的快取機制了嗎?彆著急,Maven比我們想象中的要聰明。在配置Maven的Repository的時候中有個配置項,可以配置對於SNAPSHOT版本向遠端倉庫中查詢的頻率。頻率共有四種,分別是always、daily、interval、never。當本地倉庫中存在需要的依賴專案時,always是每次都去遠端倉庫檢視是否有更新,daily是隻在第一次的時候檢視是否有更新,當天的其它時候則不會檢視;interval允許設定一個分鐘為單位的間隔時間,在這個間隔時間內只會去遠端倉庫中查詢一次,never是不會去遠端倉庫中查詢(這種就和正式版本的行為一樣了)。

Maven版本的配置方式為:

1
2
3
4
5
6
7
8
<repository>
    <id>myRepository</id>
    <url>...</url>
    <snapshots>
        <enabled>true</enabled>
        <updatePolicy>XXX</updatePolicy>
    </snapshots>
</repository>

其中updatePolicy就是那4種型別之一。如果配置間隔時間更新,可以寫作interval:XX(XX是間隔分鐘數)。daily配置是預設值。

而在Gradle,可以設定本地快取的更新策略。

1
2
3
4
5
6
configurations.all {

// check for updates every build
resolutionStrategy.cacheChangingModulesFor  0,'seconds'

}

當然也可以按照分鐘或者小時來設定.

1
2
3
4
5
configurations.all {

resolutionStrategy.cacheChangingModulesFor  10, ‘minutes'

}
1
2
3
4
5
configurations.all {

resolutionStrategy.cacheChangingModulesFor  4, ‘hours'

}

所以一般在開發模式下,我們可以頻繁的釋出SNAPSHOT版本,以便讓其它專案能實時的使用到最新的功能做聯調;當版本趨於穩定時,再釋出一個正式版本,供正式使用。當然在做正式釋出時,也要確保當前專案的依賴項中不包含對任何SNAPSHOT版本的依賴,保證正式版本的穩定性。

相關文章